C#之Equals与==


先来看一下C#关于类型的定义

C#之Equals与==

微软doc: https://docs.microsoft.com/zh-cn/dotnet/csharp/language-reference/keywords/value-types  


  1. 对于值类型而言,两者比较的都是” 内容 ”是否相同,即  值是否一样 ,很显然此时两者是划等号的。

  2. 对于引用类型来说,等号(==)比较的是两个变量的” 引用 ” 是否一样,即是引用的” 地址 ”是否相同。而对于equals来说仍然比较的是变量的  ” 内容 ” 是否一样。

   ReferenceEquals:检查同一性,看两个引用是否指向同一个对象;


      字符串的比较

            string c = "abc";
            string d = "abc";
            Console.WriteLine("Equals(c,d):           {0}", Equals(c, d));//Ture
            Console.WriteLine("ReferenceEquals(c,d):  {0}",ReferenceEquals(c, d));//Ture
            Console.WriteLine("c==d:                  {0}",c == d);//Ture
            Console.WriteLine("c.Equals(d):           {0}",c.Equals(d));//Ture

string是微软封装的一个字符串类,在内部他已经对 = = 操作符进行了重写。重写后他比较的则是两个变量的内容是否相同,重写后的 = = 操作符内部调用的即是 equals 方法,所以输出的是true


自定义的类的比较

class Person
    {
        public string name;
        public int age;
        public Person(string name,int age)
        {
            this.name = name;
            this.age = age;
        }
    }
    
            Person p1 = new Person("P1", 10);
            Person p2 = new Person("P1", 10);
            
            Console.WriteLine("Equals(p1,p2):           {0}", Equals(p1, p2));//False
            Console.WriteLine("ReferenceEquals(p1,p2):  {0}", ReferenceEquals(p1, p2));//False
            Console.WriteLine("p1==p2:                  {0}", p1 == p2);//False
            Console.WriteLine("p1.Equals(p2):           {0}", p1.Equals(p2));//False
            
            p2 = p1;
            Console.WriteLine("Equals(p1,p2):           {0}", Equals(p1, p2));//True
            Console.WriteLine("ReferenceEquals(p1,p2):  {0}", ReferenceEquals(p1, p2));//True
            Console.WriteLine("p1==p2:                  {0}", p1 == p2);//True
            Console.WriteLine("p1.Equals(p2):           {0}", p1.Equals(p2));//True


对于 p1==p2 比较的结果是false,这点是毫无疑问的,因为他俩是两个不同的变量,所以引用的地址也是不同的。

但是对于p1.Equals(p2) 返回false,可能有人会产生疑问,p1 和p2的内容是相同的啊,为什么他俩的比较结果却是为false呢?。原因就在于在Equals是Object中的一个虚方法,而person类中没有对它进行重写,因此此时调用的仍是父类中的Equals方法。但是父类是无法知道你都有哪些成员字段的,因此返回的是false。

要想让他能够比较两个变量的内容是否相同,那就应该在Person类中 重写Equals方法


public override bool Equals(object obj)
        {
            Person p = (Person)obj;
            return (this.name == p.name) && (this.age == p.age);
        }



总结:Equals比较的永远是 变量的内容 是否相同,而= =比较的则是 引用地址是否相同 (前提:此种类型内部没有对Equals 或= = 进行重写操作,否则输出可能会有不同)。string 类型是个特例,因为他的内部对这两个都进行了重写。