欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

Java用自定義的類作為HashMap的key值實例

 更新時間:2016年12月10日 09:43:20   投稿:jingxian  
下面小編就為大家?guī)硪黄狫ava用自定義的類作為HashMap的key值實例。小編覺得挺不錯的,現(xiàn)在就分享給大家,也給大家做個參考。一起跟隨小編過來看看吧

這是Java中很經(jīng)典的問題,在面試中也經(jīng)常被問起。其實很多書或者文章都提到過要重載hashCode()和equals()兩個方法才能實現(xiàn)自定義鍵在HashMap中的查找,但是為什么要這樣以及如果不這樣做會產(chǎn)生什么后果,好像很少有文章講到,所以寫這么一篇來說明下。

首先,如果我們直接用以下的Person類作為鍵,存入HashMap中,會發(fā)生發(fā)生什么情況呢?

public class Person {

  private String id;

  public Person(String id) {
    this.id = id;
  }
}
import java.util.HashMap;

public class Main {
  public static void main(String[] args) {

    HashMap<Person, String> map = new HashMap<Person, String>();

    map.put(new Person("001"), "findingsea");
    map.put(new Person("002"), "linyin");
    map.put(new Person("003"), "henrylin");
    map.put(new Person("003"), "findingsealy");

    System.out.println(map.toString());

    System.out.println(map.get(new Person("001")));
    System.out.println(map.get(new Person("002")));
    System.out.println(map.get(new Person("003")));
  }
}

那么輸出結(jié)果是什么呢?

{Person@6e4d4d5e=henrylin, Person@275cea3=findingsea, Person@15128ee5=findingsealy, Person@4513098=linyin}
null
null
null

我們可以看到,這里出現(xiàn)了兩個問題:

1.在添加的過程中,我們將key=new Person("003")的鍵值對添加了兩次,那么在期望中,HashMap中應該只存在一對這樣的鍵值對,因為key(期望中)是相同的,所以不應該重復添加,第二次添加的value="findingsealy"應該替換掉原先的value="henrylin"。但是在輸入中,我們發(fā)現(xiàn)期望中的情況并沒有出現(xiàn),而是在HashMap同時存在了value="findingsealy"和value="henrylin"的兩個鍵值對,并且它們的key值還是不相同的,這顯然是錯誤的。

2.在獲取value值時,我們分別用三個Person對象去查找,這三個對象和我們剛剛存入的三個key值(在期望中)是相同的,但是查找出的卻是三個null值,這顯然也是錯誤的。

那么,正確的方法其實在很多地方都是被描述過了,直接對Person類進行修改,重載equals和hashCode方法,修改過后的Person類如下:

 

public class Person {

  private String id;

  public Person(String id) {
    this.id = id;
  }

  @Override
  public boolean equals(Object o) {
    if (this == o) return true;
    if (o == null || getClass() != o.getClass()) return false;

    Person person = (Person) o;

    if (id != null ? !id.equals(person.id) : person.id != null) return false;

    return true;
  }

  @Override
  public int hashCode() {
    return id != null ? id.hashCode() : 0;
  }
}

那么,當我們重新執(zhí)行上述的檢驗程序時,得到的結(jié)果如下:

 

{Person@ba31=findingsea, Person@ba32=linyin, Person@ba33=findingsealy}
findingsea
linyin
findingsealy

 可以看到,之前指出的亮點錯誤都得到了改正。那么,為什么會這樣呢?

在HashMap中,查找key的比較順序為:

1.計算對象的Hash Code,看在表中是否存在。

2.檢查對應Hash Code位置中的對象和當前對象是否相等。

顯然,第一步就是要用到hashCode()方法,而第二步就是要用到equals()方法。在沒有進行重載時,在這兩步會默認調(diào)用Object類的這兩個方法,而在Object中,Hash Code的計算方法是根據(jù)對象的地址進行計算的,那兩個Person("003")的對象地址是不同的,所以它們的Hash Code也不同,自然HashMap也不會把它們當成是同一個key了。同時,在Object默認的equals()中,也是根據(jù)對象的地址進行比較,自然一個Person("003")和另一個Person("003")是不相等的。

理解了這一點,就很容易搞清楚為什么需要同時重載hashCode()和equals兩個方法了。

•重載hashCode()是為了對同一個key,能得到相同的Hash Code,這樣HashMap就可以定位到我們指定的key上。

•重載equals()是為了向HashMap表明當前對象和key上所保存的對象是相等的,這樣我們才真正地獲得了這個key所對應的這個鍵值對。

還有一個細節(jié),在Person類中對于hashCode()的重在方法為:

 

@Override
public int hashCode() {
  return id != null ? id.hashCode() : 0;
}

 這里可能有疑惑的點在于:為什么可以用String類型的變量的Hash Code作為Person類的Hash Code值呢?這樣new Person(new String("003"))和new Person(new String("003"))的Hash Code是相等的嗎?

來看看以下代碼的輸出:

System.out.println("findingsea".hashCode());
System.out.println("findingsea".hashCode());
System.out.println(new String("findingsea").hashCode());
System.out.println(new String("findingsea").hashCode());
728795174
728795174
728795174
728795174

可以看到四條語句的輸出都是相等的,很直觀的合理的猜測就是String類型也重載了hashCode()以根據(jù)字符串的內(nèi)容來返回Hash Code值,所以相同內(nèi)容的字符串具有相同的Hash Code。

同時,這也說明了一個問題:為什么在已知hashCode()相等的情況下,還需要用equals()進行比較呢?就是因為避免出現(xiàn)上述例子中的出現(xiàn)的情況,因為根據(jù)對Person類的hashCode()方法的重載實現(xiàn),Person類會直接用id這個String類型成員的Hash Code值作為自己的Hash Code值,但是很顯然的,一個Person("003")和一個String("003")是不相等的,所以在hashCode()相等的情況下,還需要用equals()進行比較。

以下例子可以作為上述說明的佐證:

System.out.println(new Person("003").hashCode()); // 47667
System.out.println(new String("003").hashCode()); // 47667

System.out.println(new Person("003").equals(new String("003"))); // false

以上這篇Java用自定義的類作為HashMap的key值實例就是小編分享給大家的全部內(nèi)容了,希望能給大家一個參考,也希望大家多多支持腳本之家。

相關(guān)文章

  • 解析Java的Spring框架的基本結(jié)構(gòu)

    解析Java的Spring框架的基本結(jié)構(gòu)

    這篇文章主要介紹了Java的Spring框架的基本結(jié)構(gòu),作者從Spring的設計角度觸發(fā)解析其基礎的架構(gòu)內(nèi)容,需要的朋友可以參考下
    2016-03-03
  • spring boot配置druid連接池的完整步驟

    spring boot配置druid連接池的完整步驟

    這篇文章主要給大家介紹了關(guān)于spring boot配置druid連接池的相關(guān)資料,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧
    2019-01-01
  • KotlinScript構(gòu)建SpringBootStarter保姆級教程

    KotlinScript構(gòu)建SpringBootStarter保姆級教程

    這篇文章主要為大家介紹了KotlinScript構(gòu)建SpringBootStarter的保姆級教程,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪
    2022-09-09
  • Java實現(xiàn)5種負載均衡算法(小結(jié))

    Java實現(xiàn)5種負載均衡算法(小結(jié))

    負載均衡是將客戶端請求訪問,通過提前約定好的規(guī)則轉(zhuǎn)發(fā)給各個server,本文主要介紹了Java實現(xiàn)5種負載均衡算法,具有一定的參考價值,感興趣的可以了解一下
    2022-06-06
  • java 將jsonarray 轉(zhuǎn)化為對應鍵值的jsonobject方法

    java 將jsonarray 轉(zhuǎn)化為對應鍵值的jsonobject方法

    下面小編就為大家分享一篇java 將jsonarray 轉(zhuǎn)化為對應鍵值的jsonobject方法,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧
    2018-03-03
  • Java實現(xiàn)的獲取和判斷文件頭信息工具類用法示例

    Java實現(xiàn)的獲取和判斷文件頭信息工具類用法示例

    這篇文章主要介紹了Java實現(xiàn)的獲取和判斷文件頭信息工具類,結(jié)合實例形式分析了Java針對文件讀取及頭信息判斷相關(guān)操作技巧,需要的朋友可以參考下
    2017-11-11
  • spring整合redis消息監(jiān)聽通知使用的實現(xiàn)示例

    spring整合redis消息監(jiān)聽通知使用的實現(xiàn)示例

    在電商系統(tǒng)中,秒殺,搶購,紅包優(yōu)惠卷等操作,一般都會設置時間限制,本文主要介紹了spring整合redis消息監(jiān)聽通知使用,具有一定的參考價值,感興趣的可以了解一下
    2021-12-12
  • java管道piped輸入流與輸出流應用場景案例分析

    java管道piped輸入流與輸出流應用場景案例分析

    這篇文章主要介紹了java管道流PipedInputStream與PipedOutputStream(輸入流與輸出流)的應用場景案例分析,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步
    2022-02-02
  • Java異常處理之try...catch...finally詳解

    Java異常處理之try...catch...finally詳解

    今天小編就為大家分享一篇關(guān)于Java異常處理之try...catch...finally詳解,小編覺得內(nèi)容挺不錯的,現(xiàn)在分享給大家,具有很好的參考價值,需要的朋友一起跟隨小編來看看吧
    2019-01-01
  • 淺談@RequestMapping注解的注意點

    淺談@RequestMapping注解的注意點

    這篇文章主要介紹了淺談@RequestMapping注解的注意點,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教
    2022-02-02

最新評論