在這里,首先得告訴大家,在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 類型是個特例,因為他的內部對這兩個都進行了重寫。