Java equals()方法使用詳解及總結(jié)
equals()
超類Object中有這個(gè)equals()方法,該方法主要用于比較兩個(gè)對(duì)象是否相等。該方法的源碼如下:
public boolean equals(Object obj) { return (this == obj); }
我們知道所有的對(duì)象都擁有標(biāo)識(shí)(內(nèi)存地址)和狀態(tài)(數(shù)據(jù)),同時(shí)“==”比較兩個(gè)對(duì)象的的內(nèi)存地址,所以說(shuō)使用Object的equals()方法是比較兩個(gè)對(duì)象的內(nèi)存地址是否相等,即若object1.equals(object2)為true,則表示equals1和equals2實(shí)際上是引用同一個(gè)對(duì)象。雖然有時(shí)候Object的equals()方法可以滿足我們一些基本的要求,但是我們必須要清楚我們很大部分時(shí)間都是進(jìn)行兩個(gè)對(duì)象的比較,這個(gè)時(shí)候Object的equals()方法就不可以了,實(shí)際上JDK中,String、Math等封裝類都對(duì)equals()方法進(jìn)行了重寫。下面是String的equals()方法:
public boolean equals(Object anObject) { if (this == anObject) { return true; } if (anObject instanceof String) { String anotherString = (String)anObject; int n = count; if (n == anotherString.count) { char v1[] = value; char v2[] = anotherString.value; int i = offset; int j = anotherString.offset; while (n-- != 0) { if (v1[i++] != v2[j++]) return false; } return true; } } return false; }
對(duì)于這個(gè)代碼段:if (v1[i++] != v2[j++])return false;我們可以非常清晰的看到String的equals()方法是進(jìn)行內(nèi)容比較,而不是引用比較。至于其他的封裝類都差不多。
在Java規(guī)范中,它對(duì)equals()方法的使用必須要遵循如下幾個(gè)規(guī)則:
equals 方法在非空對(duì)象引用上實(shí)現(xiàn)相等關(guān)系:
1、自反性:對(duì)于任何非空引用值 x,x.equals(x) 都應(yīng)返回 true。
2、對(duì)稱性:對(duì)于任何非空引用值 x 和 y,當(dāng)且僅當(dāng) y.equals(x) 返回 true 時(shí),x.equals(y) 才應(yīng)返回 true。
3、傳遞性:對(duì)于任何非空引用值 x、y 和 z,如果 x.equals(y) 返回 true,并且 y.equals(z) 返回 true,那么 x.equals(z) 應(yīng)返回 true。
4、一致性:對(duì)于任何非空引用值 x 和 y,多次調(diào)用 x.equals(y) 始終返回 true 或始終返回 false,前提是對(duì)象上 equals 比較中所用的信息沒(méi)有被修改。
5、 對(duì)于任何非空引用值 x,x.equals(null) 都應(yīng)返回 false。
對(duì)于上面幾個(gè)規(guī)則,我們?cè)谑褂玫倪^(guò)程中最好遵守,否則會(huì)出現(xiàn)意想不到的錯(cuò)誤。
在java中進(jìn)行比較,我們需要根據(jù)比較的類型來(lái)選擇合適的比較方式:
1) 對(duì)象域,使用equals方法 。
2) 類型安全的枚舉,使用equals或== 。
3) 可能為null的對(duì)象域 : 使用 == 和 equals 。
4) 數(shù)組域 : 使用 Arrays.equals 。
5) 除float和double外的原始數(shù)據(jù)類型 : 使用 == 。
6) float類型: 使用Float.foatToIntBits轉(zhuǎn)換成int類型,然后使用==。
7) double類型: 使用Double.doubleToLongBit轉(zhuǎn)換成long類型,然后使用==。
至于6)、7)為什么需要進(jìn)行轉(zhuǎn)換,我們可以參考他們相應(yīng)封裝類的equals()方法,下面的是Float類的:
public boolean equals(Object obj) { return (obj instanceof Float) && (floatToIntBits(((Float)obj).value) == floatToIntBits(value)); }
原因嘛,里面提到了兩點(diǎn):
However, there are two exceptions: If f1 and f2 both represent Float.NaN, then the equals method returns true, even though Float.NaN==Float.NaN has the value false. If <code>f1 represents +0.0f while f2 represents -0.0f, or vice versa, the equal test has the value false, even though 0.0f==-0.0f has the value true.
在equals()中使用getClass進(jìn)行類型判斷
我們?cè)诟矊慹quals()方法時(shí),一般都是推薦使用getClass來(lái)進(jìn)行類型判斷,不是使用instanceof。我們都清楚instanceof的作用是判斷其左邊對(duì)象是否為其右邊類的實(shí)例,返回boolean類型的數(shù)據(jù)??梢杂脕?lái)判斷繼承中的子類的實(shí)例是否為父類的實(shí)現(xiàn)。注意后面這句話:可以用來(lái)判斷繼承中的子類的實(shí)例是否為父類的實(shí)現(xiàn),正是這句話在作怪。我們先看如下實(shí)例(摘自《高質(zhì)量代碼 改善java程序的151個(gè)建議》)。
父類:Person
public class Person { protected String name; public String getName() { return name; } public void setName(String name) { this.name = name; } public Person(String name){ this.name = name; } public boolean equals(Object object){ if(object instanceof Person){ Person p = (Person) object; if(p.getName() == null || name == null){ return false; } else{ return name.equalsIgnoreCase(p.getName()); } } return false; } }
子類:Employee
public class Employee extends Person{ private int id; public int getId() { return id; } public void setId(int id) { this.id = id; } public Employee(String name,int id){ super(name); this.id = id; } /** * 重寫equals()方法 */ public boolean equals(Object object){ if(object instanceof Employee){ Employee e = (Employee) object; return super.equals(object) && e.getId() == id; } return false; } }
上面父類Person和子類Employee都重寫了equals(),不過(guò)Employee比父類多了一個(gè)id屬性。測(cè)試程序如下:
public class Test { public static void main(String[] args) { Employee e1 = new Employee("chenssy", 23); Employee e2 = new Employee("chenssy", 24); Person p1 = new Person("chenssy"); System.out.println(p1.equals(e1)); System.out.println(p1.equals(e2)); System.out.println(e1.equals(e2)); } }
上面定義了兩個(gè)員工和一個(gè)普通人,雖然他們同名,但是他們肯定不是同一人,所以按理來(lái)說(shuō)輸出結(jié)果應(yīng)該全部都是false,但是事與愿違,結(jié)果是:true、true、false。
對(duì)于那e1!=e2我們非常容易理解,因?yàn)樗麄儾粌H需要比較name,還需要比較id。但是p1即等于e1也等于e2,這是非常奇怪的,因?yàn)閑1、e2明明是兩個(gè)不同的類,但為什么會(huì)出現(xiàn)這個(gè)情況?首先p1.equals(e1),是調(diào)用p1的equals方法,該方法使用instanceof關(guān)鍵字來(lái)檢查e1是否為Person類,這里我們?cè)倏纯磇nstanceof:判斷其左邊對(duì)象是否為其右邊類的實(shí)例,也可以用來(lái)判斷繼承中的子類的實(shí)例是否為父類的實(shí)現(xiàn)。他們兩者存在繼承關(guān)系,肯定會(huì)返回true了,而兩者name又相同,所以結(jié)果肯定是true。
所以出現(xiàn)上面的情況就是使用了關(guān)鍵字instanceof,這是非常容易“??兆印钡?。故在覆寫equals時(shí)推薦使用getClass進(jìn)行類型判斷。而不是使用instanceof。
感謝閱讀,希望能幫助到大家,謝謝大家對(duì)本站的支持!
- Java中==符號(hào)與equals()的使用詳解(測(cè)試兩個(gè)變量是否相等)
- 詳解Java中==和equals()的區(qū)別
- Java中==和equals()的區(qū)別總結(jié)
- Java中equals()方法實(shí)例詳解
- Java中equals()方法重寫實(shí)現(xiàn)代碼
- 詳解Java中“==”與equals()的區(qū)別
- Java中的== 和equals()方法詳解與實(shí)例
- 詳解java==運(yùn)算符和equals()方法的區(qū)別
- Java自定義實(shí)現(xiàn)equals()方法過(guò)程解析
- Java中equals()方法的理解與使用方法例子
相關(guān)文章
Java學(xué)習(xí)-打印1-1000以內(nèi)的水仙花數(shù)代碼實(shí)例
這篇文章主要介紹了Java打印1-1000以內(nèi)的水仙花數(shù),文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2019-04-04Spring Boot 數(shù)據(jù)校驗(yàn)@Valid+統(tǒng)一異常處理的實(shí)現(xiàn)
這篇文章主要介紹了Spring Boot 數(shù)據(jù)校驗(yàn)@Valid+統(tǒng)一異常處理的實(shí)現(xiàn),小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2019-04-04java管道piped輸入流與輸出流應(yīng)用場(chǎng)景案例分析
這篇文章主要介紹了java管道流PipedInputStream與PipedOutputStream(輸入流與輸出流)的應(yīng)用場(chǎng)景案例分析,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步2022-02-02JDK1.7以上javaFTP上傳刪除文件的實(shí)現(xiàn)方法
下面小編就為大家分享一篇JDK1.7以上javaFTP上傳刪除文件的實(shí)現(xiàn)方法,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2017-11-11關(guān)于Java如何正確地實(shí)現(xiàn)方法重載詳解
在一個(gè)類中,可以定義多個(gè)構(gòu)造方法,這叫做方法的重載!但是關(guān)于方法重載,具有有哪些要求和細(xì)節(jié)?在今天的這篇文章中,小編給大家詳細(xì)說(shuō)說(shuō)方法重載相關(guān)的內(nèi)容,需要的朋友可以參考下2023-05-05Mybatis或Mybatis-Plus框架的xml文件中特殊符號(hào)的使用詳解
這篇文章主要介紹了Mybatis或Mybatis-Plus框架的xml文件中特殊符號(hào)的使用詳解,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2020-11-11