先来看一下C#关于类型的定义
微软doc: https://docs.microsoft.com/zh-cn/dotnet/csharp/language-reference/keywords/value-types
对于值类型而言,两者比较的都是” 内容 ”是否相同,即 值是否一样 ,很显然此时两者是划等号的。
对于引用类型来说,等号(==)比较的是两个变量的” 引用 ” 是否一样,即是引用的” 地址 ”是否相同。而对于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 类型是个特例,因为他的内部对这两个都进行了重写。