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

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


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

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

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


字符串的比较

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

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


自定义的类的比较

classPerson{publicstringname;publicintage;publicPerson(stringname,intage){this.name=name;this.age=age;}}Personp1=newPerson("P1",10);Personp2=newPerson("P1",10);Console.WriteLine("Equals(p1,p2):{0}",Equals(p1,p2));//FalseConsole.WriteLine("ReferenceEquals(p1,p2):{0}",ReferenceEquals(p1,p2));//FalseConsole.WriteLine("p1==p2:{0}",p1==p2);//FalseConsole.WriteLine("p1.Equals(p2):{0}",p1.Equals(p2));//Falsep2=p1;Console.WriteLine("Equals(p1,p2):{0}",Equals(p1,p2));//TrueConsole.WriteLine("ReferenceEquals(p1,p2):{0}",ReferenceEquals(p1,p2));//TrueConsole.WriteLine("p1==p2:{0}",p1==p2);//TrueConsole.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方法


publicoverrideboolEquals(objectobj){Personp=(Person)obj;return(this.name==p.name)&&(this.age==p.age);}



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