Android小端MD5

时间:2012-03-04 17:53:20

标签: java android endianness

我正在将一个Windows应用移植到Android上,而我遇到了一个带有字节序的问题。该应用程序从用户处获取一系列文本字段,并根据MD5生成密码。问题是当我创建字节数组以传递到MD5摘要方法时,Android应用程序上的字节采用大端格式。因此,MD5输出在两个平台之间不匹配。

我尝试使用ByteBuffer转换为little endian,然后使用ByteBuffer.get()将该值复制回字节数组。遗憾的是,这不起作用,因为它不维护订单设置..这在处理ByteBuffers时似乎是一个已知的“陷阱”。如果我比较ByteBuffer.getLong()值和windows版本中的等效值,则值匹配,但我不知道如何以正确的顺序将数组从ByteBuffer中取出。

编辑:我已经在下面添加了java和C#函数。

以下是不试图修复订单/字节顺序的java版本:

public static final long md5(final String input) {
    try {
        // Create MD5
        MessageDigest md5 = MessageDigest.getInstance("MD5");

        // Read in string as an array of bytes.
        byte[] originalBytes = input.getBytes("US-ASCII");
        byte[] encodedBytes = md5.digest(originalBytes);

        long output = 0;
        long multiplier = 1;

        // Create 64 bit integer from the MD5 hash of the input
        for (int i = 0; i < encodedBytes.length; i++) {
            output = output + encodedBytes[i] * multiplier;
            multiplier = multiplier * 255;
        }
        return output;

    } 
    catch (NoSuchAlgorithmException e) {
        e.printStackTrace();
    }
     catch (UnsupportedEncodingException e) {
        e.printStackTrace();
    }
    return 0;
}

这是C#版本

private Int64 MD5(string input)
{
  MD5CryptoServiceProvider md5 = new MD5CryptoServiceProvider();

  byte[] originalBytes = ASCIIEncoding.ASCII.GetBytes(input);
  byte[] encodedBytes = md5.ComputeHash(originalBytes);
  Int64 output = 0;
  Int64 Multiplyer = 1;
  for (int i = 0; i < encodedBytes.Length; i++)
  {
    output = output + encodedBytes[i] * Multiplyer;
    Multiplyer = Multiplyer * 255;
  }
  return output;
}

2 个答案:

答案 0 :(得分:2)

问题在于这行Java:

            output = output + encodedBytes[i] * multiplier;

与这一系列的C#代码略有不同:

    output = output + encodedBytes[i] * Multiplyer;

具体而言,encodedBytes[i]bytelong(Java)或Int64(C#)的隐式转换略有不同。

你看,在Java中,byte-128127之间的签名值,而在C#中,它是无符号< / em> 0255之间的值。因此,例如,如果encodedBytes[i]B21011 0010),则Java将其解释为-78,而C#将其解释为178。

要在Java中模拟C#解释,您可以编写如下内容:

            output = output + ((encodedBytes[i] + 256) % 256) * multiplier;

(幸运的是,Java对于整数溢出的处理与C#的“未经检查”模式相同,这显然就是你正在使用的模式; 如果你不得不模仿那么会更难以模仿。)

答案 1 :(得分:0)

MD5 standard调用128位值,而不是64位。首先,签名private Int64 MD5(string input)毫无意义。您不应该将这些转换为整数并尝试比较它们。只需传递byte[]引用并进行比较。