我正在将一个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;
}
答案 0 :(得分:2)
问题在于这行Java:
output = output + encodedBytes[i] * multiplier;
与这一系列的C#代码略有不同:
output = output + encodedBytes[i] * Multiplyer;
具体而言,encodedBytes[i]
从byte
到long
(Java)或Int64
(C#)的隐式转换略有不同。
你看,在Java中,byte
是-128
和127
之间的签名值,而在C#中,它是无符号< / em> 0
和255
之间的值。因此,例如,如果encodedBytes[i]
是B2
(1011 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[]
引用并进行比较。