Java集合Set的簡單使用解析
Set的簡單介紹
- Set接口是Collection的子接口,Set接口相較于Collection接口沒有提供額外的方法。
- Set 集合不允許包含相同的元素,如果試把兩個相同的元素加入同一個 Set 集合中,則添加操作失敗。
- Set集合支持的遍歷方式和Collection集合一樣:foreach和Iterator。
- Set的常用實現(xiàn)類有:HashSet、TreeSet、LinkedHashSet。
Set主要實現(xiàn)類:HashSet
HashSet概述
- HashSet 是 Set 接口的主要實現(xiàn)類,大多數(shù)時候使用 Set 集合時都使用這個實現(xiàn)類。
- HashSet 按 Hash 算法來存儲集合中的元素,因此具有很好的存儲、查找、刪除性能。
- HashSet 具有以下特點:
- 不能保證元素的排列順序
- HashSet 不是線程安全的
- 集合元素可以是 null
- HashSet 集合判斷兩個元素相等的標準:兩個對象通過 hashCode() 方法得到的哈希值相等,并且兩個對象的 equals() 方法返回值為true。
- 對于存放在Set容器中的對象,對應的類一定要重寫hashCode()和equals(Object obj)方法,以實現(xiàn)對象相等規(guī)則。即:“相等的對象必須具有相等的hash散列碼”。
- HashSet集合中元素的無序性,不等同于隨機性。這里的無序性與元素的添加位置有關(guān)。具體來說:我們在添加每一個元素到數(shù)組中時,具體的存儲位置是由元素的hashCode()調(diào)用后返回的hash值決定的。導致在數(shù)組中每個元素不是依次緊密存放的,表現(xiàn)出一定的無序性。
HashSet中添加元素的過程
第1步:當向 HashSet 集合中存入一個元素時,HashSet 會調(diào)用該對象的 hashCode() 方法得到該對象的 hashCode值,然后根據(jù) hashCode值,通過某個散列函數(shù)決定該對象在 HashSet 底層數(shù)組中的存儲位置。
第2步:如果要在數(shù)組中存儲的位置上沒有元素,則直接添加成功。
第3步:如果要在數(shù)組中存儲的位置上有元素,則繼續(xù)比較:
- 如果兩個元素的hashCode值不相等,則添加成功;(hashCode值不相等,兩個元素必不相同)
- 如果兩個元素的hashCode()值相等,則會繼續(xù)調(diào)用equals()方法:
- 如果equals()方法結(jié)果為false,則添加成功。
- 如果equals()方法結(jié)果為true,則添加失敗。
第2步添加成功,元素會保存在底層數(shù)組中。
第3步兩種添加成功的操作,由于該底層數(shù)組的位置已經(jīng)有元素了,則會通過鏈表的方式繼續(xù)鏈接,存儲。
重寫 hashCode() 方法的基本原則
- 在程序運行時,同一個對象多次調(diào)用 hashCode() 方法應該返回相同的值。
- 當兩個對象的 equals() 方法比較返回 true 時,這兩個對象的 hashCode() 方法的返回值也應相等。
- 對象中用作 equals() 方法比較的 Field,都應該用來計算 hashCode 值。
注意:如果兩個元素的 equals() 方法返回 true,但它們的 hashCode() 返回值不相等,hashSet 將會把它們存儲在不同的位置,但依然可以添加成功。(這是經(jīng)典的錯誤,標準的零分)
重寫equals()方法的基本原則
重寫equals方法的時候一般都需要同時復寫hashCode方法。通常參與計算hashCode的對象的屬性也應該參與到equals()中進行計算。
推薦:開發(fā)中直接調(diào)用Eclipse/IDEA里的快捷鍵自動重寫equals()和hashCode()方法即可。
Set實現(xiàn)類之二:LinkedHashSet
LinkedHashSet 是 HashSet 的子類,不允許集合元素重復。
LinkedHashSet 根據(jù)元素的 hashCode 值來決定元素的存儲位置,但它同時使用雙向鏈表維護元素的次序,這使得元素看起來是以添加順序保存的。
LinkedHashSet插入性能略低于 HashSet,但在迭代訪問 Set 里的全部元素時有很好的性能。
Set實現(xiàn)類之三:TreeSet
- TreeSet 是 SortedSet 接口的實現(xiàn)類,TreeSet 可以按照添加的元素的指定的屬性的大小順序進行遍歷。
- TreeSet底層使用紅黑樹結(jié)構(gòu)存儲數(shù)據(jù)
新增的方法如下: (了解)
- Comparator comparator()
- Object first()
- Object last()
- Object lower(Object e)
- Object higher(Object e)
- SortedSet subSet(fromElement, toElement)
- SortedSet headSet(toElement)
- SortedSet tailSet(fromElement)
TreeSet特點:不允許重復、實現(xiàn)排序(自然排序或定制排序)
- TreeSet 兩種排序方法:自然排序和定制排序。默認情況下,TreeSet 采用自然排序。
自然排序:TreeSet 會調(diào)用集合元素的 compareTo(Object obj) 方法來比較元素之間的大小關(guān)系,然后將集合元素按升序(默認情況)排列。
如果試圖把一個對象添加到 TreeSet 時,則該對象的類必須實現(xiàn) Comparable 接口。
實現(xiàn) Comparable 的類必須實現(xiàn) compareTo(Object obj) 方法,兩個對象即通過 compareTo(Object obj) 方法的返回值來比較大小。
定制排序:如果元素所屬的類沒有實現(xiàn)Comparable接口,或不希望按照升序(默認情況)的方式排列元素或希望按照其它屬性大小進行排序,則考慮使用定制排序。定制排序,通過Comparator接口來實現(xiàn)。需要重寫compare(T o1,T o2)方法。
利用int compare(T o1,T o2)方法,比較o1和o2的大?。喝绻椒ǚ祷卣麛?shù),則表示o1大于o2;如果返回0,表示相等;返回負整數(shù),表示o1小于o2。
要實現(xiàn)定制排序,需要將實現(xiàn)Comparator接口的實例作為形參傳遞給TreeSet的構(gòu)造器。
因為只有相同類的兩個實例才會比較大小,所以向 TreeSet 中添加的應該是同一個類的對象。
對于 TreeSet 集合而言,它判斷兩個對象是否相等的唯一標準是:兩個對象通過 compareTo(Object obj) 或compare(Object o1,Object o2)方法比較返回值。返回值為0,則認為兩個對象相等。
到此這篇關(guān)于Java集合Set的簡單使用解析的文章就介紹到這了,更多相關(guān)Set的簡單使用內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Java下3中XML解析 DOM方式、SAX方式和StAX方式
目前我知道的JAVA解析XML的方式有:DOM, SAX, StAX;如果選用這幾種,感覺還是有點麻煩;如果使用:JAXB(Java Architecture for XML Binding),個人覺得太方便了2013-04-04Spring @Primary和@Qualifier注解原理解析
這篇文章主要介紹了Spring @Primary和@Qualifier注解原理解析,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友可以參考下2020-04-04springboot實現(xiàn)公眾號接收回復消息和超過5秒被動回復消息
本文主要介紹了springboot實現(xiàn)公眾號接收回復消息和超過5秒被動回復消息,文中通過示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們可以參考一下2022-05-05