Соответствие вывода Java SHA2 и вывода MySQL SHA2

Когда я воспроизвожу хэш SHA2 с помощью следующего кода:

MessageDigest digest = MessageDigest.getInstance("SHA-256");
digest.digest("A".getBytes("UTF-8"));

он дает мне массив байтов, а именно: 85,-102,-22,-48,-126,100,-43,121,93,57,9,113,-116,-35,5,-85,-44,-107,114, -24,79,-27,85,-112,-18,-13,26,-120,-96,-113,-33,-3

Но когда я воспроизвожу тот же хэш через MySQL, он дает мне строку: 5cfe2cddbb9940fb4d8505e25ea77e763a0077693dbb01b1a6aa94f2.

Как я могу преобразовать результат Java, чтобы сравнить его с результатом MySQL?


person markAnthopins    schedule 11.11.2012    source источник
comment
Используйте средство форматирования строк для преобразования массива байтов в шестнадцатеричную строку.   -  person    schedule 11.11.2012
comment
Что вы имеете в виду под: it gives me a byte array, which is:85-102-22-48-126100-... как byte[] может иметь значения больше 127?   -  person jlordo    schedule 11.11.2012
comment
выходной формат может вызвать недопонимание. Вот значения, разделенные запятой: 85,-102,-22,-48,-126,100,-43,121,93,57,9,113,-116,-35,5,-85,-44,-107,114,- 24,79,-27,85,-112,-18,-13,26,-120,-96,-113,-33,-3   -  person markAnthopins    schedule 11.11.2012
comment
преобразование в шестнадцатеричную строку также не дает такого же значения.   -  person markAnthopins    schedule 11.11.2012
comment
Ах хорошо. Итак, 85 (десятичное) равно 55 шестнадцатеричному, ваш хеш MySQL начинается с 5c, вы уверены, что оба используют один и тот же алгоритм?   -  person jlordo    schedule 11.11.2012
comment
Вот результат: я думал, что это одно и то же: MySQL: SHA2('A',224) Java: (SHA-256) Очевидно, они не были одинаковыми. Так что это должно быть: MySQL: SHA2('A',256) Java: (SHA-256) Также это было полезно: stackoverflow.com/questions/4680661/   -  person markAnthopins    schedule 11.11.2012


Ответы (2)


Сначала проверьте результат своей БД, похоже, что ваш первоначальный хэш на самом деле является SHA-224, а не SHA-256:

mysql> SELECT SHA2("A", 224);
+----------------------------------------------------------+
| SHA2("A", 224)                                           |
+----------------------------------------------------------+
| 5cfe2cddbb9940fb4d8505e25ea77e763a0077693dbb01b1a6aa94f2 |
+----------------------------------------------------------+
1 row in set (0.00 sec)

Вместо:

mysql> SELECT SHA2("A", 256);
+------------------------------------------------------------------+
| SHA2("A", 256)                                                   |
+------------------------------------------------------------------+
| 559aead08264d5795d3909718cdd05abd49572e84fe55590eef31a88a08fdffd |
+------------------------------------------------------------------+
1 row in set (0.06 sec)

Оттуда вы на правильном пути, вам просто нужно преобразовать вывод byte[] в шестнадцатеричную строку.

import java.security.MessageDigest;

public class TestSHA256 {

    public static void main(String[] args) {
        try {
            MessageDigest digest = MessageDigest.getInstance("SHA-256");
            byte[] hash = digest.digest("A".getBytes("UTF-8"));

            StringBuilder hexString = new StringBuilder();
            for (int i: hash) {
                hexString.append(Integer.toHexString(0XFF & i));
            }
            System.out.println(hexString);
        } catch (Exception e) {
            e.printStackTrace();
        }
    }
}

Вывод:

559aead08264d5795d399718cdd5abd49572e84fe55590eef31a88a08fdffd
person jsuhre    schedule 11.11.2012

Integer.toHexString(0XFF & i) необходимо заменить на String.format("%02x", 0XFF & i), иначе будет выведено только 1 символ, а ожидается 2 символа.

person Maciej M.    schedule 05.05.2016