關於重寫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