使用 C# 比较两个对象是否相等的7个方法总结

简介: 比较对象是编程中的一项基本技能,在实际业务中经常碰到,比如在ERP系统中,企业的信息非常重要,每一次更新,都需要比较记录更新前后企业的信息,直接比较通常只能告诉我们它们是否指向同一个内存地址,那我们应该怎么办呢?分享 7 个方法给你!

Coding-81.png

前言

在现实中的编程生活里,我们时常遇到一个棘手的问题:如何比较两个相同类型的对象是否 "相等",比如在 ERP 系统中,企业的信息非常重要,每一次更新维护,都需要系统自动地详细记录更新前后企业不一致的信息、更新时间和更新人等等。

但是,直接比较通常只能告诉我们它们是否指向同一个内存地址,而不能告诉我们它们的内容是否一致,所以即使两个相同类型的对象的值都一致,程序还是偏偏说:"对不起,他们没关系!" 那我们应该怎么办呢?别急,接下来我们一起探索。

在 C# 中,要比较两个对象实例是否相等,有一些常用的方法。

比如实现 IEquatable<T> 接口、重写 Object.Equals 方法,或使用自定义比较逻辑等等。

以下是 7 种常用的方法:

1. 实现 IEquatable 接口

实现 IEquatable<T> 接口是一个好习惯,就像在你的小屋里挂上一个 "欢迎光临" 的牌子,让外界知道你准备好接受比较了,实现这个接口之后,你还可以创建更强类型的比较方法。

public class Test : IEquatable<Test>
{
   
    public int Id {
    get; set; }
    public string Name {
    get; set; }

    public bool Equals(Test other)
    {
   
        if (other == null)
            return false;

        return this.Id == other.Id && this.Name == other.Name;
    }

    public override bool Equals(object obj)
    {
   
        return Equals(obj as Test);
    }

    public override int GetHashCode()
    {
   
        return (Id, Name).GetHashCode(); // 可以使用 C# 7.3 中的元组
    }
}

2. 重写 Object.Equals 和 GetHashCode 方法

这是老派的做法,但依然有效。

当你重写了 EqualsGetHashCode 方法后,你就能告诉对象们: "嘿,你们俩是一样的!" 。

如果你使用的是 .NET8 或更高版本时,你还可以使用 源生成器 (source generators) 特性帮助自动生成 Equals, GetHashCode 等常用方法,编程工作更轻松!

public class Test
{
   
    public int Id {
    get; set; }
    public string Name {
    get; set; }

    public override bool Equals(object obj)
    {
   
        if (obj is Test other)
        {
   
            return this.Id == other.Id && this.Name == other.Name;
        }
        return false;
    }

    public override int GetHashCode()
    {
   
        return (Id, Name).GetHashCode();
    }
}

3. 重载 == 操作符

有时候,我们更倾向于使用 == 操作符来比较对象,为了让你的代码看起来更加自然,不妨重载一下这个操作符吧!

注意,重载 == 操作符时,通常也要重载 != 操作符

public class Test
{
   
    public int Id {
    get; set; }
    public string Name {
    get; set; }

    public static bool operator ==(Test left, Test right)
    {
   
        if (left is null && right is null) return true;
        if (left is null || right is null) return false;
        return left.Equals(right);
    }

    public static bool operator !=(Test left, Test right)
    {
   
        return !(left == right);
    }

    public override bool Equals(object obj)
    {
   
        return obj is Test other && Equals(other);
    }

    public override int GetHashCode()
    {
   
        return (Id, Name).GetHashCode();
    }
}

4. 利用匿名函数或 Lambda 表达式

如果你只是偶尔需要比较两个对象,Lambda 表达式是个不错的选择,简单又直接,快速搞定一切:

Test A = new Test {
    Id = 1, Name = "Test" };
Test B = new Test {
    Id = 1, Name = "Test" };

bool areEqual = A.Id == B.Id && A.Name == B.Name; // 手动比较属性

5. 序列化为 Json 字符串再比较

如果你需要快速比较两个复杂对象,可以考虑把它们序列化成 JSON 字符串然后进行比较,这种方法虽然不是直接的对象比较方法,但简单粗暴,在某些情况下能大显身手。

序列化推荐使用 System.Text.Json(.NET 6 及之后的版本)或 Newtonsoft.Json(第三方库)这两个库。

using System.Text.Json;

var jsonA = JsonSerializer.Serialize(A);
var jsonB = JsonSerializer.Serialize(B);

bool areEqual = jsonA == jsonB; // 比较 JSON 字符串

需要注意的是,这种方法对于复杂类型或含有大量嵌套对象的情况,性能和效率可能比较差。

6. 直接比较

虽然说大多数情况,两个相同类型的对象之间不能直接比较,但 record 类型是个例外。

如果你使用的是 .NET 6 及之后的版本,并且对象的类型是 record 类型,那么恭喜你!

因为记录类型默认实现了 EqualsGetHashCode 方法,并且提供了 ==!= 运算符重载,使得比较变得非常简单

public record Test(int Id,string Name);

// 创建两个 Test 对象
Test a = new Test(1, "test");
Test b = new Test(1, "test");

// 比较两个对象是否相等
bool areEqual = a == b; // 返回 true

7. 使用第三方库

最后,当然不能错过各种强大的第三方库,它们就像是你编程生活中的 "金牌助手" ,如果你懒得自己动手造轮子,它们会是你的最佳选择,推荐几个我常用的库:

  • EqualityComparer,一个泛型类,可以用于比较两个对象的相等性,例如:

      bool areEqual = EqualityComparer<Test>.Default.Equals(A, B);
    
  • Objects Comparer,它允许逐个成员递归地比较对象,并为某些属性、字段或类型定义自定义比较逻辑,例如:

      var comparer = new ObjectsComparer<Test>;
    
      var test1 = new Test(1, "test");
      var test2 = new Test(1, "test"); 
    
      var isEqual = comparer.Compare(test1, test2); // 比较两个对象
      ......
    
  • Compare.NET Objects,可以更详细地获取两个对象之间的差异,并记录具体的差别,例如:

      var test1 = new Test(1, "test");
      var test2 = new Test(1, "test");
    
      var propertyCount = 2;
      CompareLogic compareLogic = new CompareLogic()
      {
         
          Config = new ComparisonConfig()
          {
         
              MaxDifferences = propertyCount //MaxDifferences的默认值是1
          } 
    
      };
    
      bool result = compareLogic.Compare(test1, test2).AreEqual;
      Console.Write(result);
    

以上只是一些简单的例子,小伙伴们可以到这些第三方库的官网解锁更多使用姿势!

总结

比较对象是编程中的一项基本技能,它帮助我们维护数据的和谐,提升业务逻辑的效率,确保应用程序的正常运行,希望这 7 个方法能够让你在面对比较对象的任务时游刃有余!

记得在比较之前处理好 null 的情况,以免出现空引用异常哦!祝你编程愉快!

往期精彩

  1. C# 静态类,高手不想让你知道的 15 个真相
  2. 封装一个 C# 范围判断函数,从此告别重复编写范围判断代码的烦恼
  3. 用 C# Stopwatch 计时,让代码性能飞起来!
  4. 轻装上阵,Visual Studio LocalDB:.NET 程序员的本地数据库神器
  5. 封装一个C#万能基础数据类型转换器,一招解决所有基础类型转换烦恼
  6. 闲话 .NET(7):.NET Core 能淘汰 .NET FrameWork 吗?
  7. 常用的 4 种 ORM 框架(EF Core,SqlSugar,FreeSql,Dapper)对比总结
  8. C# AutoMapper 10个常用方法总结
  9. C# 7个方法比较两个对象是否相等
  10. C# 去掉字符串最后一个字符的 4 种方法

我是老杨,一个执着于编程乐趣、至今奋斗在一线的 10年+ 资深研发老鸟,是软件项目管理师,也是快乐的程序猿,持续免费分享全栈实用编程技巧、项目管理经验和职场成长心得,欢迎关注老杨的公众号(名称:代码掌控者),更多技术干货等你来!

相关文章
|
3月前
|
开发框架 .NET 程序员
C# 去掉字符串最后一个字符的 4 种方法
在实际业务中,我们经常会遇到在循环中拼接字符串的场景,循环结束之后拼接得到的字符串的最后一个字符往往需要去掉,看看 C# 提供了哪4种方法可以高效去掉字符串的最后一个字符
349 0
|
2月前
|
编译器 C#
C#多态概述:通过继承实现的不同对象调用相同的方法,表现出不同的行为
C#多态概述:通过继承实现的不同对象调用相同的方法,表现出不同的行为
125 65
|
1月前
|
C# UED SEO
C# 异步方法async / await任务超时处理
通过使用 `Task.WhenAny`和 `Task.Delay`方法,您可以在C#中有效地实现异步任务的超时处理机制。这种方法允许您在指定时间内等待任务完成,并在任务超时时采取适当的措施,如抛出异常或执行备用操作。希望本文提供的详细解释和代码示例能帮助您在实际项目中更好地处理异步任务超时问题,提升应用程序的可靠性和用户体验。
69 3
|
2月前
|
存储 C#
【C#】大批量判断文件是否存在的两种方法效率对比
【C#】大批量判断文件是否存在的两种方法效率对比
51 1
|
2月前
|
C#
C#的方法的参数传递
C#的方法的参数传递
25 0
|
2月前
|
数据可视化 程序员 C#
C#中windows应用窗体程序的输入输出方法实例
C#中windows应用窗体程序的输入输出方法实例
55 0
|
3月前
|
C#
C#一分钟浅谈:Lambda 表达式和匿名方法
本文详细介绍了C#编程中的Lambda表达式与匿名方法,两者均可用于定义无名函数,使代码更简洁易维护。文章通过基础概念讲解和示例对比,展示了各自语法特点,如Lambda表达式的`(parameters) =&gt; expression`形式及匿名方法的`delegate(parameters)`结构。并通过实例演示了两者的应用差异,强调了在使用Lambda时应注意闭包问题及其解决策略,推荐优先使用Lambda表达式以增强代码可读性。
49 8
|
3月前
|
C# 数据安全/隐私保护
C# 一分钟浅谈:类与对象的概念理解
【9月更文挑战第2天】本文从零开始详细介绍了C#中的类与对象概念。类作为一种自定义数据类型,定义了对象的属性和方法;对象则是类的实例,拥有独立的状态。通过具体代码示例,如定义 `Person` 类及其实例化过程,帮助读者更好地理解和应用这两个核心概念。此外,还总结了常见的问题及解决方法,为编写高质量的面向对象程序奠定基础。
28 2
|
4月前
|
图形学 C# 开发者
全面掌握Unity游戏开发核心技术:C#脚本编程从入门到精通——详解生命周期方法、事件处理与面向对象设计,助你打造高效稳定的互动娱乐体验
【8月更文挑战第31天】Unity 是一款强大的游戏开发平台,支持多种编程语言,其中 C# 最为常用。本文介绍 C# 在 Unity 中的应用,涵盖脚本生命周期、常用函数、事件处理及面向对象编程等核心概念。通过具体示例,展示如何编写有效的 C# 脚本,包括 Start、Update 和 LateUpdate 等生命周期方法,以及碰撞检测和类继承等高级技巧,帮助开发者掌握 Unity 脚本编程基础,提升游戏开发效率。
102 0
|
1月前
|
C# 开发者
C# 一分钟浅谈:Code Contracts 与契约编程
【10月更文挑战第26天】本文介绍了 C# 中的 Code Contracts,这是一个强大的工具,用于通过契约编程增强代码的健壮性和可维护性。文章从基本概念入手,详细讲解了前置条件、后置条件和对象不变量的使用方法,并通过具体代码示例进行了说明。同时,文章还探讨了常见的问题和易错点,如忘记启用静态检查、过度依赖契约和性能影响,并提供了相应的解决建议。希望读者能通过本文更好地理解和应用 Code Contracts。
35 3