BigInteger.toString 方法正在删除前导 0

2022-09-01 10:47:37

我正在尝试使用MessageDigest生成MD5总和。我有以下代码。

byte[] md5sum = digest.digest();
BigInteger bigInt = new BigInteger(1, md5sum);
output = bigInt.toString(16);

这返回的不是 32 个字符的字符串,而是 31 个字符的字符串8611c0b0832bce5a19ceee626a403a7

预期字符串为08611c0b0832bce5a19ceee626a403a7

输出中缺少前导 0。

我尝试了另一种方法

byte[] md5sum = digest.digest();
output = new String(Hex.encodeHex(md5sum));

并且输出与预期一样。

我检查了文档和整数.toString根据它进行转换

使用 Character.forDigit 提供的位数到字符映射,并在适当时在前面附加减号。

和在 Character.forDigit methos 中

如果 0 <=数字基数<,则 digit 参数有效。

有人能告诉我两种方法是如何不同的,以及为什么删除前导0?


答案 1

我个人会避免使用将二进制数据转换为文本。这并不是它真正存在的目的,即使它可以用于此目的。有大量的代码可以将a转换为其十六进制表示形式 - 例如,使用Apache Commons编解码器或简单的单一方法:BigIntegerbyte[]

private static final char[] HEX_DIGITS = "0123456789ABCDEF".toCharArray();
public static String toHex(byte[] data) {
    char[] chars = new char[data.length * 2];
    for (int i = 0; i < data.length; i++) {
        chars[i * 2] = HEX_DIGITS[(data[i] >> 4) & 0xf];
        chars[i * 2 + 1] = HEX_DIGITS[data[i] & 0xf];
    }
    return new String(chars);
}

答案 2

String.format(“%064X”, new BigInteger(1, ByteArray);

哪里

  1. 0 - 零前导符号
  2. 64 - 字符串长度
  3. X - 大写
  4. ByteArray - 要对其运行此操作的 ByteArray