IEqualityComparer用于自定义对象比较逻辑,通过实现Equals和GetHashCode方法,可在Dictionary、HashSet、LINQ等场景中按业务规则判断相等性,如Person类按Name和Age判断,避免默认引用或值比较的局限。

IEqualityComparer
该接口包含两个方法:
- bool Equals(T x, T y):判断两个对象是否相等。
- int GetHashCode(T obj):为对象生成一个哈希码,用于提高基于哈希的集合性能。
为什么要实现 IEqualityComparer?
默认情况下,引用类型使用引用相等性(即两个变量指向同一内存地址),值类型则逐字段比较。但很多时候你需要按业务规则比较对象,比如两个 Person 对象只要姓名和年龄相同就算“相等”,这时就需要自定义比较逻辑。如何为自定义类型实现 IEqualityComparer
以一个简单的 Person 类为例:public class Person
{
public string Name { get; set; }
public int Age { get; set; }
}
现在我们想让两个 Person 对象在 Name 和 Age 相同时被视为相等。可以创建一个实现 IEqualityComparer
public class PersonComparer : IEqualityComparer{ public bool Equals(Person x, Person y) { if (x == null && y == null) return true; if (x == null || y == null) return false; return x.Name == y.Name && x.Age == y.Age; } public int GetHashCode(Person obj) { if (obj == null) return 0; return HashCode.Combine(obj.Name, obj.Age); }}
如何使用自定义的比较器?
一旦实现了比较器,就可以在多种场景中使用它。1. 用在 Dictionary 中作为键:
var people = new Dictionary(new PersonComparer()); people[new Person { Name = "Alice", Age = 30 }] = "Engineer"; 2. 去除 List 中的重复项:
var list = new List{ new Person { Name = "Alice", Age = 30 }, new Person { Name = "Alice", Age = 30 } }; var distinct = list.Distinct(new PersonComparer()).ToList(); // 只保留一个
3. 用于 HashSet:
var set = new HashSet(new PersonComparer()); set.Add(new Person { Name = "Bob", Age = 25 }); set.Add(new Person { Name = "Bob", Age = 25 }); // 不会重复添加 小技巧:静态实例与泛型重用
为了避免频繁创建比较器实例,可以将其设计为单例:public class PersonComparer : IEqualityComparer{ public static readonly PersonComparer Instance = new PersonComparer(); private PersonComparer() { } public bool Equals(Person x, Person y) { ... } public int GetHashCode(Person obj) { ... }}
使用时:
var distinct = list.Distinct(PersonComparer.Instance).ToList();基本上就这些。实现 IEqualityComparer
能让你灵活控制对象的“相等”标准,是编写健壮集合操作代码的重要技能。










