Java對象比較之equals與hashCode詳解
前言
equals 方法和 hashCode 方法是 Object 類中的兩個基礎(chǔ)方法,它們共同協(xié)作來判斷兩個對象是否相等。
一、equals方法
Object 類中的 equals 方法用于檢測一個對象是否等于另外一個對象。在 Object 類中,這個方法將判斷兩個對象是否具有相同的引用。如果兩個對象具有相同的引用,它們一定是相等的。?
equals的源碼如下:
public boolean equals(Object obj) { return (this == obj); }
通過上述源碼和 equals 的定義我們可以看出,在大多數(shù)情況來說,equals 的判斷是沒有什么意義的!例如,使用 Object 中的 equals 比較兩個自定義的對象是否相等,這就完全沒有意義(因為無論對象是否相等,結(jié)果都是 false)。
通過以下示例,就可以說明這個問題:
public class EqualsMyClassExample { public static void main(String[] args) { Person u1 = new Person(); u1.setName("Java"); u1.setAge(18); Person u2 = new Person(); u1.setName("Java"); u1.setAge(18); // 打印 equals 結(jié)果 System.out.println("equals 結(jié)果:" + u1.equals(u2)); } } class Person { private String name; private int age; public String getName() { return name; } public void setName(String name) { this.name = name; } public int getAge() { return age; } public void setAge(int age) { this.age = age; } }
上述的結(jié)果為false。
所以,通常情況下,我們要判斷兩個對象是否相等,一定要重寫equals方法。
二、hashCode
1.什么是hashCode
hashCode翻譯為中文是散列碼,它是由對象推導出的一個整型值,并且這個值為任意整數(shù),包括正數(shù)或負數(shù)。?
需要注意的是:散列碼是沒有規(guī)律的。如果 x 和 y 是兩個不同的對象,x.hashCode() 與 y.hashCode() 基本上不會相同;但如果 a 和 b 相等,則 a.hashCode() 一定等于 b.hashCode()。
hashCode在Object中的源碼:
public native int hashCode();
從上述源碼可以看到,Object 中的 hashCode 調(diào)用了一個(native)本地方法,返回了一個 int 類型的整數(shù),當然,這個整數(shù)可能是正數(shù)也可能是負數(shù)。
2.hashCode的使用
相等值的hashCode一定相等
public class HashCodeExample { public static void main(String[] args) { String s1 = "Hello"; String s2 = "Hello"; String s3 = "Java"; System.out.println("s1 hashCode:" + s1.hashCode()); System.out.println("s2 hashCode:" + s2.hashCode()); System.out.println("s3 hashCode:" + s3.hashCode()); } }
程序運行結(jié)果:
不同的值 hashCode也可能相等的情況
public class HashCodeExample { public static void main(String[] args) { String s1 = "Aa"; String s2 = "BB"; System.out.println("s1 hashCode:" + s1.hashCode()); System.out.println("s2 hashCode:" + s2.hashCode()); } }
程序運行結(jié)果:
三、為什么hashCode和equals要一起重寫
下面,以set集合的使用為例子,來解釋這個問題
1.Set集合的正常使用
正常用法如下:
import java.util.HashSet; import java.util.Set; public class HashCodeExample { public static void main(String[] args) { Set<String> set = new HashSet(); set.add("Java"); set.add("Java"); set.add("MySQL"); set.add("MySQL"); set.add("Redis"); System.out.println("Set 集合長度:" + set.size()); System.out.println(); // 打印 Set 中的所有元素 set.forEach(d -> System.out.println(d)); } }
程序輸出結(jié)果:
從上述結(jié)果可以看出,重復的數(shù)據(jù)已經(jīng)被 Set 集合“合并”了,這也是 Set 集合最大的特點:去重。
2.Set集合的”異常“
然而,如果我們在 Set 集合中存儲的是,只重寫了 equals 方法的自定義對象時,Set的功能似乎就失效了。如下代碼所示:
import java.util.HashSet; import java.util.Objects; import java.util.Set; public class EqualsExample { public static void main(String[] args) { // 對象 1 Persion p1 = new Persion(); p1.setName("Java"); p1.setAge(18); // 對象 2 Persion p2 = new Persion(); p2.setName("Java"); p2.setAge(18); // 創(chuàng)建 Set 集合 Set<Persion> set = new HashSet<Persion>(); set.add(p1); set.add(p2); // 打印 Set 中的所有數(shù)據(jù) set.forEach(p -> { System.out.println(p); }); } } class Persion { private String name; private int age; // 只重寫了 equals 方法 @Override public boolean equals(Object o) { if (this == o) return true; // 引用相等返回 true // 如果等于 null,或者對象類型不同返回 false if (o == null || getClass() != o.getClass()) return false; // 強轉(zhuǎn)為自定義 Persion 類型 Persion persion = (Persion) o; // 如果 age 和 name 都相等,就返回 true return age == persion.age && Objects.equals(name, persion.name); } public String getName() { return name; } public void setName(String name) { this.name = name; } public int getAge() { return age; } public void setAge(int age) { this.age = age; } @Override public String toString() { return "Persion{" + "name='" + name + '\'' + ", age=" + age + '}'; } }
運行結(jié)果:
從上述代碼和上述圖片可以看出,即使兩個對象是相等的,Set 集合竟然沒有將二者進行去重與合并。這就是重寫了 equals 方法,但沒有重寫 hashCode 方法的問題所在。
3.解決”異常“
為了解決上面的問題,我們嘗試在重寫 equals 方法時,把 hashCode 方法也一起重寫了,實現(xiàn)代碼如下:
import java.util.HashSet; import java.util.Objects; import java.util.Set; public class EqualsToListExample { public static void main(String[] args) { // 對象 1 Persion p1 = new Persion(); p1.setName("Java"); p1.setAge(18); // 對象 2 Persion p2 = new Persion(); p2.setName("Java"); p2.setAge(18); // 創(chuàng)建 Set 對象 Set<Persion> set = new HashSet<Persion>(); set.add(p1); set.add(p2); // 打印 Set 中的所有數(shù)據(jù) set.forEach(p -> { System.out.println(p); }); } } class Persion { private String name; private int age; @Override public boolean equals(Object o) { if (this == o) return true; // 引用相等返回 true // 如果等于 null,或者對象類型不同返回 false if (o == null || getClass() != o.getClass()) return false; // 強轉(zhuǎn)為自定義 Persion 類型 Persion persion = (Persion) o; // 如果 age 和 name 都相等,就返回 true return age == persion.age && Objects.equals(name, persion.name); } @Override public int hashCode() { // 對比 name 和 age 是否相等 return Objects.hash(name, age); } public String getName() { return name; } public void setName(String name) { this.name = name; } public int getAge() { return age; } public void setAge(int age) { this.age = age; } @Override public String toString() { return "Persion{" + "name='" + name + '\'' + ", age=" + age + '}'; } }
程序運行結(jié)果:
通過上述現(xiàn)象發(fā)現(xiàn),重寫了hashCod方法后,Set集合就恢復正常了。
4.原因分析
出現(xiàn)以上問題的原因是,如果只重寫了 equals 方法,那么默認情況下,Set 進行去重操作時,會先判斷兩個對象的 hashCode 是否相同,此時因為沒有重寫 hashCode 方法,所以會直接執(zhí)行 Object 中的 hashCode 方法,而 Object 中的 hashCode 方法對比的是兩個不同引用地址的對象,所以結(jié)果是 false,那么 equals 方法就不用執(zhí)行了,直接返回的結(jié)果就是 false:兩個對象不是相等的,于是就在 Set 集合中插入了兩個相同的對象。
但是,如果在重寫 equals 方法時,也重寫了 hashCode 方法,那么在執(zhí)行判斷時會去執(zhí)行重寫的 hashCode 方法,此時對比的是兩個對象的所有屬性的 hashCode 是否相同,于是調(diào)用 hashCode 返回的結(jié)果就是 true,再去調(diào)用 equals 方法,發(fā)現(xiàn)兩個對象確實是相等的,于是就返回 true 了,因此 Set 集合就不會存儲兩個一模一樣的數(shù)據(jù)了,于是整個程序的執(zhí)行就正常了。
總結(jié)
hashCode 和 equals 兩個方法是用來協(xié)同判斷兩個對象是否相等的,采用這種方式的原因是可以提高程序插入和查詢的速度,如果在重寫 equals 時,不重寫 hashCode,就會導致在某些場景下,例如將兩個相等的自定義對象存儲在 Set 集合時,就會出現(xiàn)程序執(zhí)行的異常,為了保證程序的正常執(zhí)行,所以我們就需要在重寫 equals 時,也一并重寫 hashCode 方法才行。
到此這篇關(guān)于Java對象比較之equals與hashCode詳解的文章就介紹到這了,更多相關(guān)equals與hashCode詳解內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Mybatis-Plus中分頁插件PaginationInterceptor的使用
我們在開發(fā)的過程中,經(jīng)常會遇到分頁操作,本文主要介紹了Mybatis-Plus中分頁插件PaginationInterceptor的使用,文中通過示例代碼介紹的非常詳細,需要的朋友們下面隨著小編來一起學習學習吧2023-06-06SpringBoot 設(shè)置傳入?yún)?shù)非必要的操作
這篇文章主要介紹了SpringBoot 設(shè)置傳入?yún)?shù)非必要的操作,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧2021-02-02如何使用stream從List對象中獲取某列數(shù)據(jù)
這篇文章主要介紹了如何使用stream從List對象中獲取某列數(shù)據(jù)問題,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2022-12-12