您好,登錄后才能下訂單哦!
(1)LinkedHashSet的底層使用什么存儲元素?
(2)LinkedHashSet與HashSet有什么不同?
(3)LinkedHashSet是有序的嗎?
(4)LinkedHashSet支持按元素訪問順序排序嗎?
上一節我們說HashSet中的元素是無序的,那么有沒有什么辦法保證Set中的元素是有序的呢?
答案是當然可以。
我們今天的主角LinkedHashSet就有這個功能,它是怎么實現有序的呢?讓我們來一起學習吧。
LinkedHashSet繼承自HashSet,讓我們直接上源碼來看看它們有什么不同。
package java.util;
// LinkedHashSet繼承自HashSet
public class LinkedHashSet<E>
extends HashSet<E>
implements Set<E>, Cloneable, java.io.Serializable {
private static final long serialVersionUID = -2851667679971038690L;
// 傳入容量和裝載因子
public LinkedHashSet(int initialCapacity, float loadFactor) {
super(initialCapacity, loadFactor, true);
}
// 只傳入容量, 裝載因子默認為0.75
public LinkedHashSet(int initialCapacity) {
super(initialCapacity, .75f, true);
}
// 使用默認容量16, 默認裝載因子0.75
public LinkedHashSet() {
super(16, .75f, true);
}
// 將集合c中的所有元素添加到LinkedHashSet中
// 好奇怪, 這里計算容量的方式又變了
// HashSet中使用的是Math.max((int) (c.size()/.75f) + 1, 16)
// 這一點有點不得其解, 是作者偷懶?
public LinkedHashSet(Collection<? extends E> c) {
super(Math.max(2*c.size(), 11), .75f, true);
addAll(c);
}
// 可分割的迭代器, 主要用于多線程并行迭代處理時使用
@Override
public Spliterator<E> spliterator() {
return Spliterators.spliterator(this, Spliterator.DISTINCT | Spliterator.ORDERED);
}
}
完了,結束了,就這么多,這是全部源碼了,真的。
可以看到,LinkedHashSet中一共提供了5個方法,其中4個是構造方法,還有一個是迭代器。
4個構造方法都是調用父類的super(initialCapacity, loadFactor, true);
這個方法。
這個方法長什么樣呢?
還記得我們上一節說過一個不是public的構造方法嗎?就是它。
// HashSet的構造方法
HashSet(int initialCapacity, float loadFactor, boolean dummy) {
map = new LinkedHashMap<>(initialCapacity, loadFactor);
}
如上所示,這個構造方法里面使用了LinkedHashMap來初始化HashSet中的map。
現在這個邏輯應該很清晰了,LinkedHashSet繼承自HashSet,它的添加、刪除、查詢等方法都是直接用的HashSet的,唯一的不同就是它使用LinkedHashMap存儲元素。
那么,開篇那幾個問題是否能回答了呢?
(1)LinkedHashSet的底層使用LinkedHashMap存儲元素。
(2)LinkedHashSet是有序的,它是按照插入的順序排序的。
通過上面的學習,我們知道LinkedHashSet底層使用LinkedHashMap存儲元素,而LinkedHashMap是支持按元素訪問順序遍歷元素的,也就是可以用來實現LRU的,還記得嗎?傳送門【死磕 java集合之LinkedHashMap源碼分析】
那么,LinkedHashSet支持按元素訪問順序排序嗎?
讓我們一起來分析下。
首先,LinkedHashSet所有的構造方法都是調用HashSet的同一個構造方法,如下:
// HashSet的構造方法
HashSet(int initialCapacity, float loadFactor, boolean dummy) {
map = new LinkedHashMap<>(initialCapacity, loadFactor);
}
然后,通過調用LinkedHashMap的構造方法初始化map,如下所示:
public LinkedHashMap(int initialCapacity, float loadFactor) {
super(initialCapacity, loadFactor);
accessOrder = false;
}
可以看到,這里把accessOrder寫死為false了。
所以,LinkedHashSet是不支持按訪問順序對元素排序的,只能按插入順序排序。
歡迎關注我的公眾號“彤哥讀源碼”,查看更多源碼系列文章, 與彤哥一起暢游源碼的海洋。
免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。