7

我正在编写从 jpg 文件创建散列的 ac# 例程。如果我将一个字节数组传递给我的 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");

这会产生“哈希相同”。

我错过了什么?

4

1 回答 1

17

你没有倒带你的 MemoryStreams,所以哈希是从一个空的字节序列计算出来的。采用

ms1.Position = 0;
ms2.Position = 0;

打电话后Save

另一个注意事项:不要GetBuffer以这种方式使用。使用ToArraywhich 将为您提供与流长度相同大小的字节数组 -GetBuffer返回原始缓冲区,该缓冲区(通常)会有一些填充,您不想意外使用。GetBuffer当然,如果你确保只使用它的相关部分,你就可以使用它——这避免了创建数据的新副本。

于 2009-11-11T14:07:16.637 回答