在C#中比較兩個(gè)對(duì)象是否相等,通常有四種主要方式:
1、ReferenceEquals(object o1, object o2):
Person zs = new Person("ZhangSan",28);
Person ls = new Person("LiSi",20);
Person zsCopy = zs;
//ReferenceEquals比較兩個(gè)對(duì)象的引用,同為null是返回true
Console.WriteLine("ReferenceEquals(zs, ls) : " + ReferenceEquals(zs, ls)); // False
Console.WriteLine("ReferenceEquals(zs, zsCopy) : " + ReferenceEquals(zs, zsCopy)); //True
Console.WriteLine("ReferenceEquals(ls, zsCopy) : " + ReferenceEquals(ls, zsCopy)); //Fasle
Console.WriteLine("ReferenceEquals(null, null) : " + ReferenceEquals(null, null)); //True
Console.WriteLine("ReferenceEquals(zs, null) : " + ReferenceEquals(zs, null)); //False
Console.WriteLine("ReferenceEquals(1, 1) : " + ReferenceEquals(1, 1)); //False
2、Equals(object o):
Person zs = new Person("ZhangSan",28);
Person ls = new Person("LiSi",20);
Person zsCopy = zs;
//虛擬Equals方法,比較引用,可以重寫比較值
Console.WriteLine("zs.Equals(ls) : " + zs.Equals(ls)); //Fasle
Console.WriteLine("zs.Equals(zsCopy) : " + zs.Equals(zsCopy)); //True
Console.WriteLine("ls.Equals(zsCopy) : " + ls.Equals(zsCopy)); //False
Console.WriteLine("zs.Equals(null) : " + zs.Equals(null)); //False
Console.WriteLine("1.Equals(1) : " + 1.Equals(1)); //True
3、Equals(object o1, object o2)
Person zs = new Person("ZhangSan",28);
Person ls = new Person("LiSi",20);
Person zsCopy = zs;
//靜態(tài)Equals方法先比較兩個(gè)對(duì)象是否為空,如果都不為空,則調(diào)用虛擬Equals方法
Console.WriteLine("Person.Equals(zs, ls) : " + Person.Equals(zs, ls)); //False
Console.WriteLine("Person.Equals(zs, zsCopy) : " + Person.Equals(zs, zsCopy)); //True
Console.WriteLine("Person.Equals(ls, zsCopy) : " + Person.Equals(ls, zsCopy)); //False
Console.WriteLine("Person.Equals(null, null) : " + Person.Equals(null, null)); //True
Console.WriteLine("Person.Equals(zs, null) : " + Person.Equals(zs, null)); //Fasle
Console.WriteLine("Person.Equals(1, 1) : " + Person.Equals(1, 1)); //True
4、object1 == object2:
比較運(yùn)算符:可以重寫;對(duì)于引用類型,默認(rèn)是比較引用的(System.String除外)來源,對(duì)于值類型默認(rèn)比較值;
對(duì)于自定義的結(jié)構(gòu),如果不顯示重載operator方法,則無法使用。
由于在C#中要求比較運(yùn)算符必須成對(duì)重載,重載==運(yùn)算符的同時(shí)也必須重載!=運(yùn)算符,否則也會(huì)產(chǎn)生編譯錯(cuò)誤。
如果要重載的話,運(yùn)算符"=="、"!=" 與 Equals方法、GetHashCode方法應(yīng)該同時(shí)被重載,因?yàn)樗麄儜?yīng)該保持同樣的相等邏輯。但不要再==中調(diào)用Equals,最好是在Equals中調(diào)用==。
Person zs = new Person("ZhangSan",28);
Person ls = new Person("LiSi",20);
Person zsCopy = zs;
String str1 = "Hello";
String str2 = "Hello";
String str3 = str1;
//比較運(yùn)算符比較引用,可以重寫
Console.WriteLine("zs == ls : " + (zs == ls)); //False
Console.WriteLine("zs == zsCopy : " + (zs == zsCopy)); //True
Console.WriteLine("ls == zsCopy : " + (ls == zsCopy)); //False
Console.WriteLine("zs == null : " + (zs == null)); //False
Console.WriteLine("1 == 1 : " + (1 == 1)); //True
Console.WriteLine("str1 == str2 :" + (str1 == str2));//True
Console.WriteLine("str1 == str3 :" + (str1 == str3));//True
Console.WriteLine("str2 == str3 :" + (str2 == str3));//True
Equals(object o)與==的區(qū)別 1)==操作符判斷的是堆棧中的值,Equlas判斷的是堆中的值。
- 值類型存儲(chǔ)在棧上,故用==判斷是直接判斷其值是否相等,因?yàn)橹殿愋筒淮嬖诙阎械臄?shù)據(jù),因此值類型的Equals也是判斷數(shù)據(jù)。即,對(duì)于值類型而言,==與Equals相同,均是判斷其值是否相等。
- 對(duì)于引用類型而言,其棧中存儲(chǔ)的是對(duì)象的地址,那么==就是比較兩個(gè)地址是否相等,即是否指向同一個(gè)對(duì)象;Equals則是比較兩個(gè)對(duì)象在堆中的數(shù)據(jù)是否一樣,即兩個(gè)引用類型是否是對(duì)同一個(gè)對(duì)象的引用。
- String類型雖然是引用類型,但是對(duì)String對(duì)象的賦值卻按照值類型操作
- 對(duì)str2初始化的時(shí)候,并沒有重新開辟內(nèi)存,而是直接將其地址指向str1的內(nèi)容“hello”。這樣一來,string類型雖然是引用類型,但是其==操作和Equals操作都是一樣的,均比較值是否相等。
4、與GetHashCode()的關(guān)系
- 若兩對(duì)象Equals相等,那么其GetHashCode()必定相等;但是反過來,若GetHashCode()相等,那么這兩個(gè)對(duì)象Equals方法比較結(jié)果不一定相同。(為了獲取最佳性能,hash函數(shù)為對(duì)象內(nèi)容生成的數(shù)字是隨機(jī)分布的,這就意味著,內(nèi)容不同的對(duì)象,有可能生成的數(shù)字是一樣的,但可以認(rèn)為這種概率非常小)。
該文章在 2024/1/22 12:38:45 編輯過