LinkedHashMap原理和底層實現
1.概述
在使用HashMap的時候,可能會遇到需要按照當時put的順序來進行雜湊表的遍歷。通過上篇對HashMap的瞭解,我們知道HashMap中不存在儲存順序的機制。本篇文章要介紹的LinkedHashMap專為此特性而生。在LinkedHashMap中可以保持兩種順序,分別是插入順序和訪問順序,這個是可以在LinkedHashMap的初始化方法中進行指定的。相對於訪問順序,按照插入順序進行編排被使用到的場景更多一些,所以預設是按照插入順序進行編排。
看一下實際的執行效果,測試程式碼如下:
public static void main(String[] args) { Map<String, String> test = new LinkedHashMap<String, String>(9); test.put("化學","93"); test.put("數學","98"); test.put("生物","92"); test.put("英語","97"); test.put("物理","94"); test.put("歷史","96"); test.put("語文","99"); test.put("地理","95"); for (Map.Entry entry : test.entrySet()) { System.out.println(entry.getKey().toString() + ":" + entry.getValue().toString()); } }
執行結果如下圖所示,可以看到,輸出的順序與插入的順序是一致的。
2.原理
- 在LinkedHashMap中,是通過雙聯表的結構來維護節點的順序的。上文中的程式,實際上在記憶體中的情況如下圖所示,每個節點都進行了雙向的連線,維持插入的順序(預設)。
head
指向第一個插入的節點,tail
指向最後一個節點。 - LinkedHashMap是HashMap的親兒子,直接繼承HashMap類。LinkedHashMap中的節點元素為
Entry<K,V>
,直接繼承HashMap.Node<K,V>
。UML類圖關係如下:
3.原始碼分析
3.1 節點構造方法
剛剛看LinkedHashMap的實現的時候有個疑問。LinkedHashMap繼承HashMap,HashMap中的陣列是Node<K,V>[]
Entry<K,V>
繼承Node<K,V>[]
,但是在LinkedHashMap中並沒有找到新建節點的方法。仔細研究之後發現,在HashMap類的put
方法中,新建節點是使用的newNode
方法。而在LinkedHashMap沒有重寫父類的put
方法,而是重寫了newNode
方法來構建自己的節點物件。
HashMap中的newNode
方法:
Node<K,V> newNode(int hash, K key, V value, Node<K,V> next) { return new Node<>(hash, key, value, next); }
LinkedHashMap中的newNode
方法:
Node<K,V> newNode(int hash, K key, V value, Node<K,V> e) {
LinkedHashMap.Entry<K,V> p =
new LinkedHashMap.Entry<K,V>(hash, key, value, e);
linkNodeLast(p);
return p;
}
3.2 put方法
在LinkedHashMap類使用的仍然是父類HashMap的put方法,所以插入節點物件的流程基本一致。不同的是,LinkedHashMap重寫了afterNodeInsertion
和afterNodeAccess
方法。
afterNodeInsertion
方法用於移除連結串列中的最舊的節點物件,也就是連結串列頭部的物件。但是在JDK1.8版本中,可以看到removeEldestEntry
一直返回false
,所以該方法並不生效。如果存在特定的需求,比如連結串列中長度固定,並保持最新的N的節點資料,可以通過重寫該方法來進行實現。
void afterNodeInsertion(boolean evict) { // possibly remove eldest
LinkedHashMap.Entry<K,V> first;
if (evict && (first = head) != null && removeEldestEntry(first)) {
K key = first.key;
removeNode(hash(key), key, null, false, true);
}
}
protected boolean removeEldestEntry(Map.Entry<K,V> eldest) {
return false;
}
afterNodeAccess
方法實現的邏輯,是把入參的節點放置在連結串列的尾部。
void afterNodeAccess(Node<K,V> e) { // move node to last
LinkedHashMap.Entry<K,V> last;
if (accessOrder && (last = tail) != e) {
LinkedHashMap.Entry<K,V> p =
(LinkedHashMap.Entry<K,V>)e, b = p.before, a = p.after;
p.after = null;
if (b == null)
head = a;
else
b.after = a;
if (a != null)
a.before = b;
else
last = b;
if (last == null)
head = p;
else {
p.before = last;
last.after = p;
}
tail = p;
++modCount;
}
}
3.3 get方法
LinkedHashMap中的get方法與父類HashMap處理邏輯相似,不同之處在於增加了一處連結串列更新的邏輯。如果LinkedHashMap中存在要尋找的節點,那麼判斷如果設定了accessOrder
,則在返回值之前,將該節點移動到對應桶中連結串列的尾部。
public V get(Object key) {
Node<K,V> e;
if ((e = getNode(hash(key), key)) == null)
return null;
if (accessOrder)
afterNodeAccess(e);
return e.value;
}
3.4 remove方法
LinkedHashMap重寫了afterNodeRemoval
方法,用於在刪除節點的時候,調整雙鏈表的結構。
void afterNodeRemoval(Node<K,V> e) { // unlink
LinkedHashMap.Entry<K,V> p =
(LinkedHashMap.Entry<K,V>)e, b = p.before, a = p.after;
p.before = p.after = null;
if (b == null)
head = a;
else
b.after = a;
if (a == null)
tail = b;
else
a.before = b;
}
4.小結
LinkedHashMap相對於HashMap,增加了雙鏈表的結果(即節點中增加了前後指標),其他處理邏輯與HashMap一致,同樣也沒有鎖保護,多執行緒使用存在風險。
作者:道可
連結:https://www.imooc.com/article/23169
來源:慕課網
本文原創釋出於慕課網 ,轉載請註明出處,謝謝合作