我正在编写一个从jpg文件创建哈希的c#例程.如果我将一个字节数组传递给我的SHA512对象然后我得到了预期的行为,但是,如果我传入一个内存流,这两个文件总是散列到相同的值.
例1:
SHA512 mySHA512 = SHA512.Create(); Image img1 = Image.FromFile(@"d:\img1.jpg"); Image img2 = Image.FromFile(@"d:\img2.jpg"); MemoryStream ms1 = new MemoryStream(); MemoryStream ms2 = new MemoryStream(); img1.Save(ms1,ImageFormat.Jpeg); byte[] buf1 = ms1.GetBuffer(); byte[] hash1 = mySHA512.ComputeHash(buf1); img2.Save(ms2,ImageFormat.Jpeg); byte[] buf2 = ms2.GetBuffer(); byte[] hash2 = mySHA512.ComputeHash(buf2); if (Convert.ToBase64String(hash1) == Convert.ToBase64String(hash2)) MessageBox.Show("Hashed the same"); else MessageBox.Show("Different hashes");
这会产生“不同的哈希”.但ComputeHash方法的一个重载需要一个流对象,我宁愿使用它.当我做:
SHA512 mySHA512 = SHA512.Create(); Image img1 = Image.FromFile(@"d:\img1.jpg"); Image img2 = Image.FromFile(@"d:\img2.jpg"); MemoryStream ms1 = new MemoryStream(); MemoryStream ms2 = new MemoryStream(); img1.Save(ms1,ImageFormat.Jpeg); byte[] hash1 = mySHA512.ComputeHash(ms1); img2.Save(ms2,ImageFormat.Jpeg); byte[] hash2 = mySHA512.ComputeHash(ms2); if (Convert.ToBase64String(hash1) == Convert.ToBase64String(hash2)) MessageBox.Show("Hashed the same"); else MessageBox.Show("Different hashes");
这产生了“哈希相同”.
这里发生了什么,我失踪了?
解决方法
你没有倒退你的MemoryStreams,所以哈希是从一个空的字节序列计算出来的.使用
ms1.Position = 0; ms2.Position = 0;
在致电Save之后.
还有一点需要注意:不要以这种方式使用GetBuffer.使用ToArray将为您提供与流的长度相同的字节数组 – GetBuffer返回原始缓冲区,该缓冲区将(通常)具有一些填充,您不会意外使用它.您可以使用GetBuffer,如果您确定只使用它的相关部分,当然 – 这可以避免创建数据的新副本.