gpt4 book ai didi

c# - SHA1Managed.ComputeHash 偶尔在不同的服务器上不同

转载 作者:太空狗 更新时间:2023-10-29 20:08:00 24 4
gpt4 key购买 nike

背景(您可以跳过此部分)

我有大量数据(大约 3 MB)需要在数百台机器上保持最新。一些机器运行 C#,一些运行 Java。数据可能随时更改,需要在几分钟内传播给客户端。数据以 Json 格式从 4 个负载平衡服务器传送。这 4 台服务器运行的是 ASP.NET 4.0 以及 Mvc 3 和 C# 4.0。

在 4 个服务器上运行的代码有一个散列算法,该算法对 Json 响应进行散列,然后将散列转换为字符串。这个散列被提供给客户端。然后,每隔几分钟,客户端就会使用哈希值对服务器执行 ping 操作,如果哈希值已过期,则返回新的 Json 对象。如果哈希仍然是最新的,则返回带有空主体的 304。

偶尔 4 个盒子生成的哈希值在各个盒子之间不一致,这意味着客户端不断下载数据(每个请求都可能到达不同的服务器)。

代码片段

这是用于生成哈希的代码。

internal static HashAlgorithm Hasher { get; set; }
...
Hasher = new SHA1Managed();
...
Convert.ToBase64String(Hasher.ComputeHash(Encoding.ASCII.GetBytes(jsonString)));

为了尝试调试问题,我将其拆分如下:

Prehash = PreHashBuilder.ToString();
ASCIIBytes = Encoding.ASCII.GetBytes(Prehash);
HashedBytes = Hasher.ComputeHash(ASCIIBytes);
Hash = Convert.ToBase64String(HashedBytes);

然后我添加了一条路由,它吐出上述值并使用 Beyond Compare 来比较差异。

字节数组转换为字符串格式以供 BeyondCompare 使用:

private static string GetString(byte[] bytes)
{
StringBuilder sb = new StringBuilder();
foreach (byte b in bytes)
{
sb.Append(b);
}
return sb.ToString();
}

如您所见,字节数组按字节顺序显示。它没有“转换”。

问题

我发现 Prehash 和 ASCIIBytes 值相同,但 HashedBytes 值不同 - 这意味着 Hash 也不同。

我多次重启了 4 个服务器上的 IIS 网站,当它们有不同的哈希值时,比较了 BeyondCompare 中的值。在每种情况下,不同的是“HashedBytes”值(SHA1Managed.ComputeHash(...) 的结果)

问题

我做错了什么? ComputeHash 函数的输入是相同的。 SHA1Managed 机器依赖吗?这不会成功,因为 4 台机器有一半的时间具有相同的哈希值。

我已经搜索过 StackOverFlow 和 Bing,但找不到其他人遇到这个问题。我能找到的最接近的是编码有问题的人,但我想我已经证明编码不是问题。

输出

我希望不要把所有东西都转储到这里,因为它有多长,但这是我正在比较的转储的片段:

哈希值:o1ZxBaVuU6OhE6De96wJXUvmz3M=
HashedBytes:163861135165110831631611916022224717299375230207115
ASCIIBytes:1151169710310146991111094779114100101114831011141181059910147115101114118105991014611511899591151051031101171129511510111411810599101114101102101114101110991011159598979910710111010011111410010111411510111411810599101951185095117114108611041161161125847471051159897991071011101004610910211598101115116971031014699111109477911410010111483101114118105991014711510111411810599101461151189947118505911510510311011711295115101114118105991011141011021011141011109910111595989799107101110100112971211091011101161151161111141011151011141....预哈希:...

当我比较不同服务器上的两个页面时,ASCII 字节相同但 HashedByte 不同。我对字节使用的转储方法不进行任何转换,它只是按顺序转储每个字节。我可以用“.”来分隔字节。我想。

跟进我对 b.ToString(CultureInfo.InvariantCulture) 进行了更改,并将 HashAlgorithm 设为局部变量而不是静态属性。我正在等待代码部署到服务器。

最佳答案

我一直在尝试重现这个问题,但是一旦我将 SHA1Managed 属性设为局部变量而不是全局静态变量,我就无法重现。

问题出在多线程上。我的代码是线程安全的,除了我标记为静态的 SHA1Managed 类。我假设 SHA1Managed.ComputeHash 在下面是线程安全的,但显然如果标记为内部静态则不是。

重复一遍,如果标记为内部静态,SHA1Managed.ComputeHash 不是线程安全的。

MSDN 声明:

Any public static (Shared in Visual Basic) members of this type are thread safe. Any instance members are not guaranteed to be thread safe.

我不知道为什么内部静态的行为与公共(public)静态不同。

我会将@pst 标记为答案并添加评论以澄清问题,但@pst 发表了评论,因此我无法将其标记为答案。

感谢您的所有意见。

关于c# - SHA1Managed.ComputeHash 偶尔在不同的服务器上不同,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/12644257/

24 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com