溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務條款》

Java中HashSet集合怎么對自定義對象進行去重

發(fā)布時間:2021-09-18 11:25:15 來源:億速云 閱讀:507 作者:chen 欄目:編程語言

本篇內容介紹了“Java中HashSet集合怎么對自定義對象進行去重”的有關知識,在實際案例的操作過程中,不少人都會遇到這樣的困境,接下來就讓小編帶領大家學習一下如何處理這些情況吧!希望大家仔細閱讀,能夠學有所成!

Java中Set接口是Collectio的子接口,Set集合不允許包含相同的元素。如果添加相同的元素, add()會返回FALSE,  新元素不會加入。Set集合常用于元素為數(shù)字、字符串去重等,但是當元素為自定義對象類型時,Set去重是否與我們預計一致?下面將以HashSet為例,通過一系列試驗來一步步驗證。

1. 先建立一個FootBallPlayer足球運動員類

Java中HashSet集合怎么對自定義對象進行去重

2. (假設:HashSet會把屬性值全相同的對象認定為重復),為了測試HashSet對對象去重效果與猜想是否一致,我們先構建三個對象實例,其中構造兩個屬性一致的“C羅”。

Java中HashSet集合怎么對自定義對象進行去重

結果:HashSet并沒有認定兩個“C羅”對象重復,三個實例都加入到了HashSet集合中。

Java中HashSet集合怎么對自定義對象進行去重

3. 在了解HashSet如何進行去重之前,先看看HashSet是怎么實現(xiàn)的。通過查看JDK源碼發(fā)現(xiàn)HashSet內部其實是對HashMap進行操作。

Java中HashSet集合怎么對自定義對象進行去重

4. 繼續(xù)查看hashSet的add()方法,其實是調用了HashMap的put()方法

Java中HashSet集合怎么對自定義對象進行去重

5. 繼續(xù)追蹤,直到putVal()方法(重點)

Java中HashSet集合怎么對自定義對象進行去重

Java中HashSet集合怎么對自定義對象進行去重

仔細看putVal()方法,發(fā)現(xiàn)其對于新入的元素是否重復判斷依據為以下兩種

  • 判斷hash值是否相等,既通過判斷hashCode()方法

  • 判斷是否相等,通過equals()方法

6. 了解了兩個判斷條件后,我們先做一個簡單實驗,既調用Integer 、String 、Object等對象equals()方法進行對比

Java中HashSet集合怎么對自定義對象進行去重

結果發(fā)現(xiàn),自定義Object對象equals返回的值為false。接下來我們逐一看看它們的equals實現(xiàn)方式

Java中HashSet集合怎么對自定義對象進行去重

(1) Integer對象的equals實現(xiàn),通過閱讀代碼發(fā)現(xiàn)是判斷依據是值是否相等。

Java中HashSet集合怎么對自定義對象進行去重

(2) String對象的equals實現(xiàn),其判斷的依據為:先判斷引用的對象是否是同一個,再逐個對比其字符串的值

Java中HashSet集合怎么對自定義對象進行去重

(3)  而Object的判斷依據為引用的對象是否是同一個,由于上面的兩位足球運動員都是新new出來的,非同一個對象,所以equlas()返回結果為false

Java中HashSet集合怎么對自定義對象進行去重

7. 看完了equlas的實現(xiàn),接下來看看Integer String  Object的hashCode實現(xiàn)。同樣先做一個簡單的測試,調用它們的hashCode()方法計算出hash值進行對比

Java中HashSet集合怎么對自定義對象進行去重

實驗為結果兩個Object對象的hash值并不相等,接下來我們看看它們對于hashcode()的具體實現(xiàn)

Java中HashSet集合怎么對自定義對象進行去重

(1) 通過源碼發(fā)現(xiàn) Integer是通過對其value值來進運算行得到hash值。

Java中HashSet集合怎么對自定義對象進行去重

(2) String也是通過對其value值來進計算行得到hash值,所以測試中結果為true

Java中HashSet集合怎么對自定義對象進行去重

(3) 當查看Object的hashCode()方法時發(fā)現(xiàn)并無具體實現(xiàn),通過查閱資料得知,JDK8的默認hashCode的計算是交給C++實現(xiàn)的,方法是通過和當前線程有關的一個隨機數(shù)+三個確定值,運用Marsaglia's

xorshifschema隨機數(shù)算法得到的一個隨機數(shù)。所以兩個不同的對象得到的hash值便不相同,測試結果也為false。(對于Object的hashCode()這里不做深入討論,如果過深入了解的朋友也歡迎分享)

Java中HashSet集合怎么對自定義對象進行去重

8. 得知了HashSet是通過hashcode()與equals()來進行去重,且自定義Object對象的equals()和hashcode()實現(xiàn)原理,那么要實現(xiàn)HashSet按照我們期望的方式,當兩個對象所有屬性的值一致時認定為同一個對象,我們可以對FootBallPlayer類的equals()和hashcode()進行重寫,代碼如下

Java中HashSet集合怎么對自定義對象進行去重

  • hashCode() 重寫為hash值是通過對對象所有屬性的值進行運算得出。

  • equals() 重寫為先判斷引用的對象是否是同一個,再判斷對象每一個屬性值是否相等

9. 重寫完方法,我們再重新執(zhí)行一開始的程序,還是同樣的三個足球運動員實例。結果與期望相同,HashSet對“C羅”對象進行了去重處理。

Java中HashSet集合怎么對自定義對象進行去重

總結

HashSet的底層是對HashMap的操作,其去重的原理通過hashCode()與equals()方法來判斷是否重復。通過實驗發(fā)現(xiàn)自定義對象沒有成功去重的原因與JDK默認的Object對象hashCode()和equals()實現(xiàn)有關。對于自定義對象的去重,我們可以通過重寫自定義對象的hashCode()與equals()使其按照我們所想要的規(guī)則進行去重操作。

“Java中HashSet集合怎么對自定義對象進行去重”的內容就介紹到這里了,感謝大家的閱讀。如果想了解更多行業(yè)相關的知識可以關注億速云網站,小編將為大家輸出更多高質量的實用文章!

向AI問一下細節(jié)

免責聲明:本站發(fā)布的內容(圖片、視頻和文字)以原創(chuàng)、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯(lián)系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。

AI