文件的散列与校验:.NET发现之旅(五)

如何保障文档在互联网传输过程中的安全性不是这一节讨论的内容,我们要讨论的内容是如何验证文档的完整性。记得08年我接手绵阳九院的一个MOSS项目,由于对方单位知识的保密性,平时电脑启动就需要U盘解密,所以当时接管负责这个项目的时候,对方首先要求凡是通过网络传输的文档,必须提供MD5验证,所以当时就顺手写了这个工具FileChecker,今天分享给大家。
验证数据完整性,我们常用哈希算法。关于哈希算法,无论是MD5还是SHA1,最终生成的哈希值理论上都是不可逆的,主要用于保证数据的完整性,这和对称和非对称加密不同,后者是可逆的,主要是为了防窃取。由于很多网站使用MD5加密某些数据,比如密码、权限等,引起很多***想方设法的穷举破解MD5值,但多半都是白费力气,运气好的时候就撞上了,但是这个运气不是每个人都有的。一个安全的哈希算法在设计时必须满足两个要求:其一是寻找两个输入得到相同的输出值在计算上是不可行的,这就是我们通常听说的抗碰撞;其二是找一个输出,能得到给定的输入在计算上是不可行的,即不可从结果推导出它的初始状态。据说王小云教授当年就是采用碰撞破解了MD5一部分,也证明了MD5不是完美的,但至今MD5是没有被破解的。
FileChecker用.NET开发,其中核心类比较简单,就不做解释了,代码如下:
 
InBlock.gif using System;
InBlock.gif using System.Collections.Generic;
InBlock.gif using System.Text;
InBlock.gif using System.Security.Cryptography;
InBlock.gif using System.IO;
InBlock.gif
InBlock.gif namespace MyMD5Checker
InBlock.gif{
InBlock.gif         class MD5AndSHA1
InBlock.gif        {
InBlock.gif                 /// <summary>
InBlock.gif                 /// 用MD5散列文件
InBlock.gif                 /// </summary>
InBlock.gif                 /// <param name="curFileName">要散列的文件(全路径)</param>
InBlock.gif                 /// <param name="flag">散列值是否使用"-"符号,为true时使用</param>
InBlock.gif                 /// <returns></returns>
InBlock.gif                 ///    
InBlock.gif                 public string MD5File( string curFileName, bool flag)
InBlock.gif                {
InBlock.gif                         return HashFile(curFileName, "md5",flag);
InBlock.gif                }
InBlock.gif
InBlock.gif                 /// <summary>
InBlock.gif                 /// 用SHA1散列文件
InBlock.gif                 /// </summary>
InBlock.gif                 /// <param name="curFileName">要散被列的文件(全路径)</param>
InBlock.gif                 /// <param name="flag">散列值是否使用"-"符号,为true时使用</param>
InBlock.gif                 /// <returns></returns>
InBlock.gif                 ///    
InBlock.gif                 public string SHA1File( string curFileName, bool flag)
InBlock.gif                {
InBlock.gif                         return HashFile(curFileName, "sha1",flag);
InBlock.gif                }
InBlock.gif
InBlock.gif                 /// <summary>
InBlock.gif                 /// 散列文件
InBlock.gif                 /// </summary>
InBlock.gif                 /// <param name="curFileName">要被散列的文件(全路径)</param>
InBlock.gif                 /// <param name="algName">所用的散列算法名字,例如MD5,SHA1</param>
InBlock.gif                 /// <param name="flag">散列值是否使用"-"符号,为true时使用</param>
InBlock.gif                 /// <returns></returns>
InBlock.gif                 private string HashFile( string curFileName, string algName, bool flag)
InBlock.gif                {
InBlock.gif                         if (!System.IO.File.Exists(curFileName))
InBlock.gif                                 return string.Empty;
InBlock.gif                        System.IO.FileStream fs = new System.IO.FileStream(curFileName, System.IO.FileMode.Open, System.IO.FileAccess.Read, System.IO.FileShare.Read);
InBlock.gif                         byte[] hashBytes = HashData(fs, algName);
InBlock.gif                        fs.Close();
InBlock.gif                         return ByteArrayToHexString(hashBytes,flag);
InBlock.gif                }
InBlock.gif
InBlock.gif                 /// <summary>
InBlock.gif                 /// 具体的散列实现
InBlock.gif                 /// </summary>
InBlock.gif                 /// <param name="stream">当前文件的流式</param>
InBlock.gif                 /// <param name="algName">哈希算法名</param>
InBlock.gif                 /// <returns></returns>
InBlock.gif                 private byte[] HashData(System.IO.Stream stream, string algName)    
InBlock.gif                {    
InBlock.gif                        System.Security.Cryptography.HashAlgorithm algorithm;
InBlock.gif
InBlock.gif                         if ( string.Compare(algName, "sha1", true) == 0)    
InBlock.gif                        {    
InBlock.gif                                algorithm = System.Security.Cryptography.SHA1.Create();
InBlock.gif                        }    
InBlock.gif                         else    
InBlock.gif                        {    
InBlock.gif                                 if ( string.Compare(algName, "md5", true) != 0)    
InBlock.gif                                {    
InBlock.gif                                         throw new Exception( "algName只能是sha1或md5");    
InBlock.gif                                }    
InBlock.gif                                algorithm = System.Security.Cryptography.MD5.Create();
InBlock.gif                        }
InBlock.gif                         return algorithm.ComputeHash(stream);
InBlock.gif                }
InBlock.gif
InBlock.gif                 /// <summary>
InBlock.gif                 /// 调整文件散列值的返回格式
InBlock.gif                 /// </summary>
InBlock.gif                 /// <param name="buf">文件散列值的二进制格式</param>
InBlock.gif                 /// <param name="flag">散列值是否使用"-"符号,为true时使用</param>
InBlock.gif                 /// <returns></returns>
InBlock.gif                 private string ByteArrayToHexString( byte[] buf, bool flag)    
InBlock.gif                {
InBlock.gif                         if (flag)
InBlock.gif                        {
InBlock.gif                                 return BitConverter.ToString(buf);
InBlock.gif                        }
InBlock.gif                         else
InBlock.gif                        {
InBlock.gif                                 return BitConverter.ToString(buf).Replace( "-", "");    
InBlock.gif                        }
InBlock.gif                }
InBlock.gif        }
InBlock.gif}
 
FileChecker实现了用MD5和SHA1算法来散列文件和校验文件。当你在给对方传输文件之前,你先用FileChecker生成MD5和SHA1哈希值,当对方接到你的文件后,再用FileChecker验证哈希值。当然,你也可以用FileChecker校验下载的文件,只需要拿到软件提供者发布的MD5或者SHA1哈希值就OK了,很多游戏网站在下载客户端的时候,都提供了MD5校验值,便是这里所说的哈希值。FileChecker界面如下:
 
1,散列文件
 
 
2,校验文件
 
 
如何用就不介绍了,非常简单。
 
最后是这个软件的安装包,因为这个软件是用.NET开发的,如果你的电脑上没装.NET Framework 2.0 ,请去微软下载中心下载。只所以做安装包的时候没把.NET Framework 2.0打进来,是因为这个.NET Framework 2.0有22.4M,影响上传和下载的速度。所以请自行下载.NET Framework 2.0安装。
 
最后提供FileChecker x86和x64版本的下载。
 
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值