0

我正在使用以下代码为我的应用程序生成 MD5 哈希,它编码一些字符串值,然后我通过我的 Web 服务将生成的哈希发送到一些 .Net 代码,该代码读取我的 MD5 哈希并生成字符串的 MD5 他是从数据库获取然后比较它们

public static String getMD5Hash(String val) throws Exception {
        byte[] bytes = val.getBytes();

        MessageDigest m = MessageDigest.getInstance("MD5");
        byte[] digest = m.digest(bytes);
        String hash = new BigInteger(1, digest).toString(16);
        System.out.println(hash.length());
        return hash;
    }

    public static void main(String[] asd) throws Exception{
        for(int i=0;i<10;i++){
            System.out.println(getMD5Hash(i+Math.pow(10, i)+""));//for testing
            System.out.println(getMD5Hash(i+""));//for testing
        }
    }

问题是当我从 BigInteger 获得转换后的哈希时,有时生成的哈希以 0 开头,在这种情况下,Biginteger 不考虑 0,而另一方面,当 .Net 生成相同字符串的哈希时,它会使用 0 和以这种方式字符串比较返回false,例如我生成并发送的哈希码“ 102678D1922799CF9122B3F103975F1”在.Net中的哈希是“ 0102678D1922799CF9122B3F103975F1

有时它也会生成类似的代码

.Net generates  0012678D1922799CF9122B3F103975F1 and java 12678D1922799CF9122B3F103975F1 

或者

0002678D1922799CF9122B3F103975F1 and 2678D1922799CF9122B3F103975F1

我怎样才能得到这个全为 0 的代码。提前致谢。

4

2 回答 2

4

我认为没有MessageDigest比仅填充字符串更好的方法了:

// ...
StringBuilder builder = new StringBuilder();
for (int i = hash.length(); i < 64; i++) {
  builder.append('0');
}
builder.append(hash);
return builder.toString();

...但你可以使用GuavaStrings.padStart()

// ...
return Strings.padStart(hash, 64, '0');

...甚至使用Guava 的哈希库,它返回与您的函数完全相同的字符串(我刚刚检查过):

public static String getMD5Hash(String val) throws Exception {
  return Hashing.md5().hashBytes(val.getBytes()).toString();
}
于 2012-11-02T05:49:59.880 回答
3

这个问题有一些更好的生成十六进制字符串的方法:

在 Java 中,如何将字节数组转换为十六进制数字字符串,同时保持前导零?

由于您已经在使用BigInteger,这似乎是最好的解决方案:

BigInteger bi = new BigInteger(1, digest);
String hash = String.format("%0" + (digest.length << 1) + "X", bi);
于 2012-11-02T18:40:20.753 回答