您好,登錄后才能下訂單哦!
本篇內容介紹了“Java中HashSet集合怎么對自定義對象進行去重”的有關知識,在實際案例的操作過程中,不少人都會遇到這樣的困境,接下來就讓小編帶領大家學習一下如何處理這些情況吧!希望大家仔細閱讀,能夠學有所成!
Java中Set接口是Collectio的子接口,Set集合不允許包含相同的元素。如果添加相同的元素, add()會返回FALSE, 新元素不會加入。Set集合常用于元素為數字、字符串去重等,但是當元素為自定義對象類型時,Set去重是否與我們預計一致?下面將以HashSet為例,通過一系列試驗來一步步驗證。
1. 先建立一個FootBallPlayer足球運動員類
2. (假設:HashSet會把屬性值全相同的對象認定為重復),為了測試HashSet對對象去重效果與猜想是否一致,我們先構建三個對象實例,其中構造兩個屬性一致的“C羅”。
結果:HashSet并沒有認定兩個“C羅”對象重復,三個實例都加入到了HashSet集合中。
3. 在了解HashSet如何進行去重之前,先看看HashSet是怎么實現的。通過查看JDK源碼發現HashSet內部其實是對HashMap進行操作。
4. 繼續查看hashSet的add()方法,其實是調用了HashMap的put()方法
5. 繼續追蹤,直到putVal()方法(重點)
仔細看putVal()方法,發現其對于新入的元素是否重復判斷依據為以下兩種
判斷hash值是否相等,既通過判斷hashCode()方法
判斷是否相等,通過equals()方法
6. 了解了兩個判斷條件后,我們先做一個簡單實驗,既調用Integer 、String 、Object等對象equals()方法進行對比
結果發現,自定義Object對象equals返回的值為false。接下來我們逐一看看它們的equals實現方式
(1) Integer對象的equals實現,通過閱讀代碼發現是判斷依據是值是否相等。
(2) String對象的equals實現,其判斷的依據為:先判斷引用的對象是否是同一個,再逐個對比其字符串的值
(3) 而Object的判斷依據為引用的對象是否是同一個,由于上面的兩位足球運動員都是新new出來的,非同一個對象,所以equlas()返回結果為false
7. 看完了equlas的實現,接下來看看Integer String Object的hashCode實現。同樣先做一個簡單的測試,調用它們的hashCode()方法計算出hash值進行對比
實驗為結果兩個Object對象的hash值并不相等,接下來我們看看它們對于hashcode()的具體實現
(1) 通過源碼發現 Integer是通過對其value值來進運算行得到hash值。
(2) String也是通過對其value值來進計算行得到hash值,所以測試中結果為true
(3) 當查看Object的hashCode()方法時發現并無具體實現,通過查閱資料得知,JDK8的默認hashCode的計算是交給C++實現的,方法是通過和當前線程有關的一個隨機數+三個確定值,運用Marsaglia's
xorshifschema隨機數算法得到的一個隨機數。所以兩個不同的對象得到的hash值便不相同,測試結果也為false。(對于Object的hashCode()這里不做深入討論,如果過深入了解的朋友也歡迎分享)
8. 得知了HashSet是通過hashcode()與equals()來進行去重,且自定義Object對象的equals()和hashcode()實現原理,那么要實現HashSet按照我們期望的方式,當兩個對象所有屬性的值一致時認定為同一個對象,我們可以對FootBallPlayer類的equals()和hashcode()進行重寫,代碼如下
hashCode() 重寫為hash值是通過對對象所有屬性的值進行運算得出。
equals() 重寫為先判斷引用的對象是否是同一個,再判斷對象每一個屬性值是否相等
9. 重寫完方法,我們再重新執行一開始的程序,還是同樣的三個足球運動員實例。結果與期望相同,HashSet對“C羅”對象進行了去重處理。
總結
HashSet的底層是對HashMap的操作,其去重的原理通過hashCode()與equals()方法來判斷是否重復。通過實驗發現自定義對象沒有成功去重的原因與JDK默認的Object對象hashCode()和equals()實現有關。對于自定義對象的去重,我們可以通過重寫自定義對象的hashCode()與equals()使其按照我們所想要的規則進行去重操作。
“Java中HashSet集合怎么對自定義對象進行去重”的內容就介紹到這里了,感謝大家的閱讀。如果想了解更多行業相關的知識可以關注億速云網站,小編將為大家輸出更多高質量的實用文章!
免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。