目标是要得到一个文件的32位十六进制表示的MD5值,如同用WinMD5软件计算出来的结果一样。
.net提供的计算MD5值的类是位于System.Security.Cryptography命名空间下的MD5CryptoServiceProvider类。我使用其中的提供Stream对象作为参数的ComputeHash重载方法计算指定文件的MD5值。代码如下:
using System;
using System.IO;
using System.Security.Cryptography;
class App
{
static void Main()
{
string path = @"d:\temp.txt";
FileStream fs = new FileStream(path,FileMode.Open,FileAccess.Read);
MD5CryptoServiceProvider md5 = new MD5CryptoServiceProvider();
byte [] md5byte = md5.ComputeHash(fs);
foreach (byte b in md5byte)
{
Console.Write(Convert.ToString(b,16));
}
Console.ReadLine();
}
}
这样就可以显示出该文件的MD5值了。
但是这样有个问题,就是结果不一定总是32位的。原因就是使用Convert.ToString()方法在转换的时候,不会将类似0x2这样的数字转换成0x0和0x2两个字节,导致最后的结果少于32位,也就跟用WinMD5计算的结果不一致了。所以我还得手动将每一个字节拆分成两个字节,并转换成十六进制的字符串表示。方法就是对每一个字节,分别左移和右移四位,得到两个数,然后再分别转换成一个字节表示的十六进制(不知道谁有更先进的办法
)。这样得到的结果就跟用WinMD5计算得到的结果一致了,完整代码如下:using System;
using System.IO;
using System.Security.Cryptography;
class App
{
static void Main()
{
string path = @"d:\temp.txt";
FileStream fs = new FileStream(path,FileMode.Open,FileAccess.Read);
MD5CryptoServiceProvider md5 = new MD5CryptoServiceProvider();
byte [] md5byte = md5.ComputeHash(fs);
int i,j;
foreach (byte b in md5byte)
{
i = Convert.ToInt32(b);
j = i >> 4;
Console.Write(Convert.ToString(j,16));
j = ((i << 4) & 0x00ff) >> 4;
Console.Write(Convert.ToString(j,16));
}
Console.ReadLine();
}
}
简单解释一下其中的关键部分:
j = i >> 4,是将高四位提取出来,得到和原始数高四位等值的一个数
i << 4是左移四位,但是高四位仍然存在,所以就和0x00ff按位与来去掉高四位,再把结果右移四位就得到了和原始数低四位等值的一个数了
这样就将一个字节拆分成了高四位和低四位两个数了。
比较麻烦的就是获得低四位,而这其中的关键就是左移后高位数仍然存在,我开始不知道怎么办,后来问了一个网友才知道可以通过和0x00ff按位与来解决这个问题的(唉,笨死了)。