成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

面試官:重寫 Equals 時為什么一定要重寫 HashCode?

開發(fā) 前端
quals 方法和 hashCode 方法是 Object 類中的兩個基礎(chǔ)方法,它們共同協(xié)作來判斷兩個對象是否相等。為什么要這樣設(shè)計嘞?原因就出在“性能” 2 字上。

[[438320]]

本題目難度:低

常見程度:高

equals 方法和 hashCode 方法是 Object 類中的兩個基礎(chǔ)方法,它們共同協(xié)作來判斷兩個對象是否相等。為什么要這樣設(shè)計嘞?原因就出在“性能” 2 字上。

使用過 HashMap 我們就知道,通過 hash 計算之后,我們就可以直接定位出某個值存儲的位置了,那么試想一下,如果你現(xiàn)在要查詢某個值是否在集合中?如果不通過 hash 方式直接定位元素(的存儲位置),那么就只能按照集合的前后順序,一個一個的詢問比對了,而這種依次比對的效率明顯低于 hash 定位的方式。這就是 hash 以及 hashCode 存在的價值。

當我們對比兩個對象是否相等時,我們就可以先使用 hashCode 進行比較,如果比較的結(jié)果是 true,那么就可以使用 equals 再次確認兩個對象是否相等,如果比較的結(jié)果是 true,那么這兩個對象就是相等的,否則其他情況就認為兩個對象不相等。這樣就大大的提升了對象比較的效率,這也是為什么 Java 設(shè)計使用 hashCode 和 equals 協(xié)同的方式,來確認兩個對象是否相等的原因。

那為什么不直接使用 hashCode 就確定兩個對象是否相等呢?

這是因為不同對象的 hashCode 可能相同;但 hashCode 不同的對象一定不相等,所以使用 hashCode 可以起到快速初次判斷對象是否相等的作用。

但即使知道了以上基礎(chǔ)知識,依然解決不了本篇的問題,也就是:重寫 equals 時為什么一定要重寫 hashCode?要想了解這個問題的根本原因,我們還得先從這兩個方法開始說起。

1.equals 方法

Object 類中的 equals 方法用于檢測一個對象是否等于另外一個對象。在 Object 類中,這個方法將判斷兩個對象是否具有相同的引用。如果兩個對象具有相同的引用,它們一定是相等的。

equals 方法的實現(xiàn)源碼如下:

  1. public boolean equals(Object obj) { 
  2.     return (this == obj); 

通過上述源碼和 equals 的定義我們可以看出,在大多數(shù)情況來說,equals 的判斷是沒有什么意義的!例如,使用 Object 中的 equals 比較兩個自定義的對象是否相等,這就完全沒有意義(因為無論對象是否相等,結(jié)果都是 false)。

通過以下示例,就可以說明這個問題:

  1. public class EqualsMyClassExample { 
  2.     public static void main(String[] args) { 
  3.         Person u1 = new Person(); 
  4.         u1.setName("Java"); 
  5.         u1.setAge(18); 
  6.    
  7.         Person u2 = new Person(); 
  8.         u1.setName("Java"); 
  9.         u1.setAge(18); 
  10.          
  11.         // 打印 equals 結(jié)果 
  12.         System.out.println("equals 結(jié)果:" + u1.equals(u2)); 
  13.     } 
  14.  
  15. class Person { 
  16.     private String name
  17.     private int age; 
  18.     public String getName() { 
  19.         return name
  20.     } 
  21.     public void setName(String name) { 
  22.         this.name = name
  23.     } 
  24.     public int getAge() { 
  25.         return age; 
  26.     } 
  27.     public void setAge(int age) { 
  28.         this.age = age; 
  29.     } 

以上程序的執(zhí)行結(jié)果,如下圖所示:圖片因此通常情況下,我們要判斷兩個對象是否相等,一定要重寫 equals 方法,這就是為什么要重寫 equals 方法的原因。

2.hashCode 方法

hashCode 翻譯為中文是散列碼,它是由對象推導(dǎo)出的一個整型值,并且這個值為任意整數(shù),包括正數(shù)或負數(shù)。

需要注意的是:散列碼是沒有規(guī)律的。如果 x 和 y 是兩個不同的對象,x.hashCode() 與 y.hashCode() 基本上不會相同;但如果 a 和 b 相等,則 a.hashCode() 一定等于 b.hashCode()。

hashCode 在 Object 中的源碼如下:

  1. public native int hashCode(); 

從上述源碼可以看到,Object 中的 hashCode 調(diào)用了一個(native)本地方法,返回了一個 int 類型的整數(shù),當然,這個整數(shù)可能是正數(shù)也可能是負數(shù)。

hashCode 使用

相等的值 hashCode 一定相同的示例:

  1. public class HashCodeExample { 
  2.     public static void main(String[] args) { 
  3.         String s1 = "Hello"
  4.         String s2 = "Hello"
  5.         String s3 = "Java"
  6.         System.out.println("s1 hashCode:" + s1.hashCode()); 
  7.         System.out.println("s2 hashCode:" + s2.hashCode()); 
  8.         System.out.println("s3 hashCode:" + s3.hashCode()); 
  9.     } 

以上程序的執(zhí)行結(jié)果,如下圖所示:

不同的值 hashCode 也有可能相同的示例:

  1. public class HashCodeExample { 
  2.     public static void main(String[] args) { 
  3.         String s1 = "Aa"
  4.         String s2 = "BB"
  5.         System.out.println("s1 hashCode:" + s1.hashCode()); 
  6.         System.out.println("s2 hashCode:" + s2.hashCode()); 
  7.     } 

以上程序的執(zhí)行結(jié)果,如下圖所示:圖片

3.為什么要一起重寫?

接下來回到本文的主題,重寫 equals 為什么一定要重寫 hashCode?

為了解釋這個問題,我們需要從下面的這個例子入手。

3.1 Set 正常使用

Set 集合是用來保存不同對象的,相同的對象就會被 Set 合并,最終留下一份獨一無二的數(shù)據(jù)。

它的正常用法如下:

  1. import java.util.HashSet; 
  2. import java.util.Set
  3.  
  4. public class HashCodeExample { 
  5.     public static void main(String[] args) { 
  6.         Set<String> set = new HashSet(); 
  7.         set.add("Java"); 
  8.         set.add("Java"); 
  9.         set.add("MySQL"); 
  10.         set.add("MySQL"); 
  11.         set.add("Redis"); 
  12.         System.out.println("Set 集合長度:" + set.size()); 
  13.         System.out.println(); 
  14.         // 打印 Set 中的所有元素 
  15.         set.forEach(d -> System.out.println(d)); 
  16.     } 

以上程序的執(zhí)行結(jié)果,如下圖所示:

從上述結(jié)果可以看出,重復(fù)的數(shù)據(jù)已經(jīng)被 Set 集合“合并”了,這也是 Set 集合最大的特點:去重。

3.2 Set 集合的“異常”

然而,如果我們在 Set 集合中存儲的是,只重寫了 equals 方法的自定義對象時,有趣的事情就發(fā)生了,如下代碼所示:

  1. import java.util.HashSet; 
  2. import java.util.Objects; 
  3. import java.util.Set
  4.  
  5. public class EqualsExample { 
  6.     public static void main(String[] args) { 
  7.         // 對象 1 
  8.         Persion p1 = new Persion(); 
  9.         p1.setName("Java"); 
  10.         p1.setAge(18); 
  11.         // 對象 2 
  12.         Persion p2 = new Persion(); 
  13.         p2.setName("Java"); 
  14.         p2.setAge(18); 
  15.         // 創(chuàng)建 Set 集合 
  16.         Set<Persion> set = new HashSet<Persion>(); 
  17.         set.add(p1); 
  18.         set.add(p2); 
  19.         // 打印 Set 中的所有數(shù)據(jù) 
  20.         set.forEach(p -> { 
  21.             System.out.println(p); 
  22.         }); 
  23.     } 
  24.  
  25.  
  26. class Persion { 
  27.     private String name
  28.     private int age; 
  29.  
  30.     // 只重寫了 equals 方法 
  31.     @Override 
  32.     public boolean equals(Object o) { 
  33.         if (this == o) return true; // 引用相等返回 true 
  34.         // 如果等于 null,或者對象類型不同返回 false 
  35.         if (o == null || getClass() != o.getClass()) return false
  36.         // 強轉(zhuǎn)為自定義 Persion 類型 
  37.         Persion persion = (Persion) o; 
  38.         // 如果 age 和 name 都相等,就返回 true 
  39.         return age == persion.age && 
  40.                 Objects.equals(name, persion.name); 
  41.     } 
  42.   
  43.     public String getName() { 
  44.         return name
  45.     } 
  46.     public void setName(String name) { 
  47.         this.name = name
  48.     } 
  49.     public int getAge() { 
  50.         return age; 
  51.     } 
  52.     public void setAge(int age) { 
  53.         this.age = age; 
  54.     } 
  55.      
  56.      @Override 
  57.     public String toString() { 
  58.         return "Persion{" + 
  59.                 "name='" + name + '\'' + 
  60.                 ", age=" + age + 
  61.                 '}'
  62.     } 

以上程序的執(zhí)行結(jié)果,如下圖所示:圖片從上述代碼和上述圖片可以看出,即使兩個對象是相等的,Set 集合竟然沒有將二者進行去重與合并。這就是重寫了 equals 方法,但沒有重寫 hashCode 方法的問題所在。

3.3 解決“異常”

為了解決上面的問題,我們嘗試在重寫 equals 方法時,把 hashCode 方法也一起重寫了,實現(xiàn)代碼如下:

  1. import java.util.HashSet; 
  2. import java.util.Objects; 
  3. import java.util.Set
  4.  
  5. public class EqualsToListExample { 
  6.     public static void main(String[] args) { 
  7.         // 對象 1 
  8.         Persion p1 = new Persion(); 
  9.         p1.setName("Java"); 
  10.         p1.setAge(18); 
  11.         // 對象 2 
  12.         Persion p2 = new Persion(); 
  13.         p2.setName("Java"); 
  14.         p2.setAge(18); 
  15.         // 創(chuàng)建 Set 對象 
  16.         Set<Persion> set = new HashSet<Persion>(); 
  17.         set.add(p1); 
  18.         set.add(p2); 
  19.         // 打印 Set 中的所有數(shù)據(jù) 
  20.         set.forEach(p -> { 
  21.             System.out.println(p); 
  22.         }); 
  23.     } 
  24.  
  25.  
  26. class Persion { 
  27.     private String name
  28.     private int age; 
  29.  
  30.     @Override 
  31.     public boolean equals(Object o) { 
  32.         if (this == o) return true; // 引用相等返回 true 
  33.         // 如果等于 null,或者對象類型不同返回 false 
  34.         if (o == null || getClass() != o.getClass()) return false
  35.         // 強轉(zhuǎn)為自定義 Persion 類型 
  36.         Persion persion = (Persion) o; 
  37.         // 如果 age 和 name 都相等,就返回 true 
  38.         return age == persion.age && 
  39.                 Objects.equals(name, persion.name); 
  40.     } 
  41.  
  42.     @Override 
  43.     public int hashCode() { 
  44.         // 對比 name 和 age 是否相等 
  45.         return Objects.hash(name, age); 
  46.     } 
  47.      
  48.     public String getName() { 
  49.         return name
  50.     } 
  51.     public void setName(String name) { 
  52.         this.name = name
  53.     } 
  54.     public int getAge() { 
  55.         return age; 
  56.     } 
  57.     public void setAge(int age) { 
  58.         this.age = age; 
  59.     } 
  60.      
  61.     @Override 
  62.     public String toString() { 
  63.         return "Persion{" + 
  64.                 "name='" + name + '\'' + 
  65.                 ", age=" + age + 
  66.                 '}'
  67.     } 

以上程序的執(zhí)行結(jié)果,如下圖所示:

通過上述結(jié)果可以看出,當我們一起重寫了兩個方法之后,奇跡的事情又發(fā)生了,Set 集合又恢復(fù)正常了,這是為什么呢?

3.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,就會導(dǎo)致在某些場景下,例如將兩個相等的自定義對象存儲在 Set 集合時,就會出現(xiàn)程序執(zhí)行的異常,為了保證程序的正常執(zhí)行,所以我們就需要在重寫 equals 時,也一并重寫 hashCode 方法才行。

 

責任編輯:武曉燕 來源: Java面試真題解析
相關(guān)推薦

2021-12-13 09:10:48

equalshashCodeJava

2023-10-08 07:13:19

equalshashCode哈希表

2021-07-30 09:32:55

JavaEquals

2021-03-18 23:38:26

EqualsHashcode方法

2019-08-16 10:10:07

hashcodeequalsJava

2020-12-23 13:29:15

微服務(wù)架構(gòu)面試官

2021-12-30 06:59:28

方法重寫面試

2019-01-29 11:02:30

消息中間件Java互聯(lián)網(wǎng)

2022-05-26 09:24:09

volatile懶漢模式

2021-03-05 11:02:14

iOS 14.5蘋果更新

2022-07-06 13:48:24

RedisSentinel機制

2022-03-21 07:40:08

線程池Executors方式

2011-05-10 15:51:34

SEO

2023-12-25 09:03:33

MySQL索引數(shù)據(jù)庫

2023-12-06 09:10:28

JWT微服務(wù)

2020-10-24 15:50:54

Java值傳遞代碼

2021-02-19 10:02:57

HTTPSJava安全

2021-01-21 07:53:29

面試官Promis打印e

2022-04-24 09:54:24

ProxyReflect前端

2009-03-11 11:12:24

點贊
收藏

51CTO技術(shù)棧公眾號

主站蜘蛛池模板: 欧美一级在线免费 | 久久精品在线免费视频 | 给我免费的视频在线观看 | 欧美a级网站 | 欧美日韩在线观看一区 | 日韩一区二区三区在线观看视频 | 亚洲黄色一级毛片 | 亚洲三区在线观看 | 免费一区二区 | 九九久久这里只有精品 | 视频一区二区三区四区五区 | 欧美电影免费网站 | 精品国产乱码久久久久久影片 | 福利片在线看 | 精品久久一区二区 | 亚洲精品久久久一区二区三区 | 亚洲一区二区久久 | 欧美日韩精品 | av免费入口 | 狠狠色网 | 日本大香伊一区二区三区 | 欧美国产中文 | 看一级毛片视频 | 国产探花在线观看视频 | 亚洲成人精品一区 | 欧美v片| 日本精品一区二区三区四区 | 超碰成人免费 | 日韩中文字幕在线视频 | 伊人网综合在线观看 | 一区二区精品在线 | 伊人网影院 | 亚洲精品久久久一区二区三区 | 亚洲免费在线观看 | 一区日韩 | 国产成人在线视频 | 欧美日韩在线高清 | 九九久视频| 亚洲欧美日韩一区 | 欧美成人精品一区二区三区 | 欧洲精品视频一区 |