c# - HashSet.Contains 和自定义 IEqualityComparer 的意外结果

标签 c# .net json.net hashset

对于将自定义比较器与 HashSet 结合使用,我一定有某种误解。我收集了许多不同类型的数据,中间存储为 Json。为了对其进行操作,我使用了 Json.NET,特别是 JObjectJArrayJtoken

通常我会在收集时向这些内容添加一些内联元数据,并以“tbp_”为前缀。我需要知道表示为 JObject 的特定数据位是否已被收集(或未被收集)。为此,我有一个自定义 IEqualityComparer,它扩展了 Json.NET 提供的实现。它在使用提供的实现检查值是否相等之前去除元数据:

public class EntryComparer : JTokenEqualityComparer
{
    private static string _excludedPrefix = "tbp_";

    public JObject CloneForComparison(JObject obj)
    {
        var clone = obj.DeepClone() as JObject;
        var propertiesToRemove = clone
            .Properties()
            .Where(p => p.Name.StartsWith(_excludedPrefix));

        foreach (var property in propertiesToRemove)
        {
            property.Remove();
        }

        return clone;
    }

    public bool Equals(JObject obj1, JObject obj2)
    {
        return base.Equals(CloneForComparison(obj1), CloneForComparison(obj2));
    }

    public int GetHashCode(JObject obj)
    {
        return base.GetHashCode(CloneForComparison(obj));
    }
}

我使用 HashSet 来跟踪我正在操作的数据,因为我只需要知道它是否已经存在。我使用 EntryComparer 的实例初始化 HashSet。我的测试是:

public class EntryComparerTests
{
    EntryComparer comparer;
    JObject j1;
    JObject j2;

    public EntryComparerTests()
    {
        comparer = new EntryComparer();
        j1 = JObject.Parse(@"
        {
          'tbp_entry_date': '2017-03-25T21:25:53.127993-04:00',
          'from_date': '1/6/2017',
          'to_date': '2/7/2017',
          'use': '324320',
          'reading': 'act',
          'kvars': '0.00',
          'demand': '699.10',
          'bill_amt': '$28,750.75'
        }");
        j2 = JObject.Parse(@"
        {
          'tbp_entry_date': '2017-03-10T18:59:00.537745-05:00',
          'from_date': '1/6/2017',
          'to_date': '2/7/2017',
          'use': '324320',
          'reading': 'act',
          'kvars': '0.00',
          'demand': '699.10',
          'bill_amt': '$28,750.75'
        }");
    }

    [Fact]
    public void Test_Equality_Comparer_GetHashCode()
    {  
        Assert.Equal(comparer.GetHashCode(j1), comparer.GetHashCode(j2));
        Assert.Equal(true, comparer.Equals(j1, j2));
    }

    [Fact]
    public void Test_Equality_Comparer_Hashset_Contains()
    {
        var hs = new HashSet<JObject>(comparer);
        hs.Add(j1);

        Assert.Equal(true, hs.Contains(j2));
    }
}

Test_Equality_Comparer_GetHashCode() 通过,但 Test_Equality_Comparer_Hashset_Contains() 失败。 j1j2 应该被视为相等,并且根据第一次测试的结果,那么我在这里缺少什么?

最佳答案

更改类的签名:

public class EntryComparer : JTokenEqualityComparer, IEqualityComparer<JObject>

否则 GetHashCode()Equals()使用的是基类中的那些(具有不同的“签名”......基类实现了 IEqualityComparer<JToken> ,因此你的方法不会被 HashSet<> 调用)。

然后有一个属性删除的小错误:

var propertiesToRemove = clone
    .Properties()
    .Where(p => p.Name.StartsWith(_excludedPrefix))
    .ToArray();

更好的做法是“隐藏”JTokenEqualityComparer并将其设为私有(private)字段,例如:

public class EntryComparer : IEqualityComparer<JObject>
{
    private static readonly JTokenEqualityComparer _comparer = new JTokenEqualityComparer();
    private static readonly string _excludedPrefix = "tbp_";

    public static JObject CloneForComparison(JObject obj)
    {
        var clone = obj.DeepClone() as JObject;
        var propertiesToRemove = clone
            .Properties()
            .Where(p => p.Name.StartsWith(_excludedPrefix))
            .ToArray();

        foreach (var property in propertiesToRemove)
        {
            property.Remove();
        }

        return clone;
    }

    public bool Equals(JObject obj1, JObject obj2)
    {
        return _comparer.Equals(CloneForComparison(obj1), CloneForComparison(obj2));
    }

    public int GetHashCode(JObject obj)
    {
        return _comparer.GetHashCode(CloneForComparison(obj));
    }
}

关于c# - HashSet.Contains 和自定义 IEqualityComparer 的意外结果,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/43071177/

相关文章:

c# - 在 asp.net core 中需要经过身份验证的用户,但在某些操作中需要自定义策略需要自定义策略

.net - UAC - .net 的嵌入 list

c# - JsonConvert.DeserializeObject<Da​​taTable>(jsondata);强制日期时间到字符串列

c# - 使用 Json.Net 解析 forecast.io 天气数据时遇到问题

c# - 如果在实例化 DbContext 资源的 using 语句中数据库连接失败,我如何自动抛出 ApplicationException?

c# - Parallel.For 在列表中,仅在开始时按顺序对项目进行操作

c# - 什么时候应该使用Using语句?

c# - 无法手动或自动将 ILNumerics 控件添加到 VS2012 工具箱

c# - 如何获取我在SQL中输入的值

c# - 使用 JSON.NET 解析 C# 中的 Json