1. 程式人生 > >關於重寫equals與hashCode

關於重寫equals與hashCode

@Override  
public boolean equals(Object obj) {  
     if (this == obj)   //判斷物件地址是否相等,如果是就不用判斷,提高效率  
         return true;  
     if (obj == null)   //物件為空,則不往下走了  
         return false;  
     if (getClass() != obj.getClass())  //判斷兩個物件是否一樣:class july.star.equals.Student--class july.star.equals.Student  
         return false;  
    User user = (User) obj;  //向下轉型  
     /** 判斷成員變數是否一樣 */  
     if (age != user.age)     
         return false;  //判斷年齡,int型別,直接比較
     //String型別,equals()比較
     if (uname == null) {  
         if (user.uname != null)  
              return false;  
     } else if (!uname.equals(user.uname))  
         return false;  
     if (sex == null) {  
         if (user.sex != null)  
              return false;  
     } else if (!sex.equals(user.sex))  
         return false;  
     if (city == null) {  
         if (user.city != null)  
              return false;  
     } else if (!city.equals(user.city))  
         return false; 
     return true;  
}
//重寫equals要實現hachCode()  
@Override  
public int hashCode() {  
     final int prime = 31;  
     int result = 1;  
     result = prime * result + age;  
     result = prime * result + ((uname == null) ? 0 : uname.hashCode());  
     result = prime * result + ((sex == null) ? 0 : sex.hashCode());  
     result = prime * result + ((city == null) ? 0 : city.hashCode());  
     return result;  
}  

多個欄位組合作為聯合主鍵,必須實現equals和hashcode方法

equals()和hashCode()這兩個方法屬於Object類,而Object類是所有類的父類,因此所有的類都繼承了這兩個方法。其中有一些類重寫了這兩個方法。

例如:Object類的equals()方法程式碼如下: public boolean equals(Object obj) {       return (this == obj);       }  

這兩個方法都來自於Object物件,根據API文件檢視下原意。

(1)public boolean equals(Objectobj),對於任何非空引用值 x 和 y,當且僅當 x 和 y 引用同一個物件時,此方法才返回 true;

注意:當此方法被重寫時,通常有必要重寫 hashCode 方法,以維護 hashCode 方法的常規協定,該協定宣告相等物件必須具有相等的雜湊碼。

(2)public int hashCode() 返回該物件的雜湊碼值。支援該方法是為雜湊表提供一些優點

    我們知道,如果不重寫equals,那麼比較的將是物件的引用是否指向同一塊記憶體地址,重寫之後目的是為了比較兩個物件的value值是否相等。特別指出,此時,利用equals比較八大包裝物件(如int,float等)和String類(因為該類已重寫了equals和hashcode方法)物件時,預設比較的是值,在比較其它物件都是比較的引用地址。那產生了一個問題,為什麼jdk中希望我們在重寫equals時,非常有必要重寫hashcode呢?

    我的理解是hashcode是用於雜湊資料的快速存取,如利用HashSet/HashMap/Hashtable類來儲存資料時,都是根據儲存物件的hashcode值來進行判斷是否相同的。這樣如果我們對一個物件重寫了euqals,意思是隻要物件的成員變數值都相等那麼euqals就等於true,但不重寫hashcode,那麼我們再new一個新的物件,當原物件.equals(新物件)等於true時,兩者的hashcode卻是不一樣的,由此將產生了理解的不一致,如在儲存雜湊集合時(如Set類),將會儲存了兩個值一樣的物件,導致混淆,因此,就也需要重寫hashcode。

在集合類(HashMap,HashSet等)中判斷兩個物件是否相等有如下規則:

如果兩個物件雜湊值不同,那麼這兩個物件不相等。如果相同,則呼叫equals()方法判斷,如果equals()方法返回true,則這兩個物件相等,否則不相等。為了保證這種一致性,必須滿足以下兩個條件:     (1)當obj1.equals(obj2)為true時,obj1.hashCode() == obj2.hashCode()必須為true

    (2)當obj1.hashCode() == obj2.hashCode()為false時,obj1.equals(obj2)必須為false