目标是要得到一个文件的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按位与来解决这个问题的 c#获取文件md5值using System;
using System.Collections.Generic; using System.Text; using System.IO; namespace MyHash { class md5 { /// <summary> /// 实现对一个文件md5的读取,path为文件路径 /// </summary> /// <param name="path"></param> /// <returns></returns> public string md5_hash(string path) { try { FileStream get_file = new FileStream(path, FileMode.Open, FileAccess.Read, FileShare.Read); System.Security.Cryptography.MD5CryptoServiceProvider get_md5 = new System.Security.Cryptography.MD5CryptoServiceProvider(); byte[] hash_byte = get_md5.ComputeHash(get_file); string resule = System.BitConverter.ToString(hash_byte); resule = resule.Replace("-", ""); return resule; } catch (Exception e) { return e.ToString(); } } } } using System; using System.Collections.Generic; using System.Text; using System.IO; using System.Security.Cryptography; public static void getFileMD5(string path) { 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(); } |
|