在这里,首先得告诉大家,在C#中,不管是 == ,还是 Equals() 我们都是可以重写的。所以,在具体选择使用两者时,还是得根据这两个方法的具体实现逻辑来选择。
值类型的比较
对于值类型来说,两者比较的都是”内容”是否相同,即 值是否一样,很显然此时两者是划等号的。
1 static void Main(string[] args) 2 { 3 int a = 1; 4 int b = 1; 5 Console.WriteLine("== 比较结果:" + (a == b)); 6 Console.WriteLine("Eauals() 比较结果:" + a.Equals(b)); 7 Console.ReadKey(); 8 }
结论:对于值类型来说 ,两者是相同的,都是比较的变量的值。
引用类型的比较
对于引用类型来说,等号 (==) 比较的是两个变量中存储的”引用” 是否一样,即是引用的”地址”是否相同。而对于 equals 来说仍然比较的是变量的 ”内容” 是否一样
1、 字符串的比较(string)
1 static void Main(string[] args) 2 { 3 string str1 = "123"; 4 string str2 = "123"; 5 Console.WriteLine("== 比较结果:" + (str1 == str2)); 6 Console.WriteLine("Eauals() 比较结果:" + str1.Equals(str2)); 7 Console.ReadKey(); 8 }
大家可能会问 上面说的 = = 号是比较两个引用类型的变量的引用是否一致,但是上面的 str1 和 str2 是两个不同的变量,为何比较的输出是 true 呢?这就是我最开始说的,具体还是要看两个方法里面的具体实现逻辑。由于 string 类型是微软封装的一个字符串类,在内部他已经对 = = 操作符进行了重写。重写后他比较的则是两个变量的内容是否相同,请看下图:
使用 Reflector 反编译工具 查看这个方法的源码 如下:
重写后的 = = 操作符内部调用的即是 equals() 方法,所以输出的是 true。
2、 自定义的其他引用类型比较
1 class Program 2 { 3 static void Main(string[] args) 4 { 5 Person p1 = new Person("mmm"); 6 Person p2 = new Person("mmm"); 7 Console.WriteLine("== 比较结果:" + (p1 == p2)); 8 Console.WriteLine("Eauals() 比较结果:" + p1.Equals(p2)); 9 Console.ReadKey(); 10 } 11 } 12 13 class Person 14 { 15 private string name; 16 public Person(string name) 17 { 18 this.name = name; 19 } 20 }
对于 p1==p2 比较的结果是 false ,这点是毫无疑问的,因为他俩是两个不同的变量,所以引用的地址也是不同的。但是对于 p1.Equals(p2) 返回 false ,可能有人会产生疑问,p1 和p2的内容是相同的啊,为什么他俩的比较结果却是为false呢?。原因就在于在Equals()是 Object 中的一个虚方法,而 Person 类中没有对她进行重写,因此此时调用的仍是父类中的 Equals() 方法。但是父类是无法知道你都有哪些成员字段的,因此返回的是 false 。要想让他能够比较两个变量的内容是否相同,那就应该重写 Equals() 方法,代码如下:
1 class Program 2 { 3 static void Main(string[] args) 4 { 5 Person p1 = new Person("mmm"); 6 Person p2 = new Person("mmm"); 7 Console.WriteLine("== 比较结果:" + (p1 == p2)); 8 Console.WriteLine("Eauals() 比较结果:" + p1.Equals(p2)); 9 Console.ReadKey(); 10 } 11 } 12 13 class Person 14 { 15 private string name; 16 public Person(string name) 17 { 18 this.name = name; 19 } 20 21 public override bool Equals(object obj) 22 { 23 Person person = obj as Person; 24 if (this.name == person.name) 25 { 26 return true; 27 } 28 else 29 { 30 return false; 31 32 } 33 } 34 }
总结:Equals() 比较的永远是变量的内容是否相同,而 = = 比较的则是引用地址是否相同。(前提:此种类型内部没有对 Equals() 或 = = 进行重写操作,否则输出可能会有不同)
string 类型是个特例,因为他的内部对这两个都进行了重写。