摘要:如今行至于此,當(dāng)觀賞一方。由于所返回的無(wú)執(zhí)行意義。源碼閱讀總體門(mén)檻相對(duì)而言比,畢竟大多數(shù)底層都由實(shí)現(xiàn)了。比心可通過(guò)這篇文章理解創(chuàng)建一個(gè)實(shí)例過(guò)程圖工作原理往期線(xiàn)路回顧源碼一帶一路系列之源碼一帶一路系列之源碼一帶一路系列之
本文以jdk1.8中LinkedHashMap.afterNodeAccess()方法為切入點(diǎn),分析其中難理解、有價(jià)值的源碼片段(類(lèi)似源碼查看是ctrl+鼠標(biāo)左鍵的過(guò)程)。觀光線(xiàn)路圖:afterNodeAccess() --> afterNodeInsertion() --> removeEldestEntry() --> afterNodeRemoval() --> internalWriteEntries() ...
? afterNodeAccess()void afterNodeAccess(Nodee) { // move node to last LinkedHashMap.Entry last; if (accessOrder && (last = tail) != e) { LinkedHashMap.Entry p = (LinkedHashMap.Entry )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; } } //////////////////////////////// 涉及變量如下: /** * The head (eldest) of the doubly linked list. */ transient LinkedHashMap.Entry head; /** * The tail (youngest) of the doubly linked list. */ transient LinkedHashMap.Entry tail; /** * The iteration ordering method for this linked hash map: true * for access-order, false for insertion-order. * * @serial */ final boolean accessOrder;
上回在HashMap.afterNodeAccess()中說(shuō)道,“是為L(zhǎng)inkedHashMap留的后路”。如今行至于此,當(dāng)觀賞一方。首先需要了解的是LinkedHashMap相比HashMap多了有序性,由雙向鏈表(before,after)實(shí)現(xiàn)。源碼出現(xiàn)了一些全局變量:
accessOrder:true:按訪問(wèn)順序排序(LRU),false:按插入順序排序;
head、tail:存放鏈表首尾;
可見(jiàn)僅有accessOrder為true時(shí),且訪問(wèn)節(jié)點(diǎn)不等于尾節(jié)點(diǎn)時(shí),該方法才有意義。通過(guò)before、after重定向,將新訪問(wèn)節(jié)點(diǎn)鏈接為鏈表尾節(jié)點(diǎn)。
? afterNodeInsertion()void afterNodeInsertion(boolean evict) { // possibly remove eldest LinkedHashMap.Entryfirst; if (evict && (first = head) != null && removeEldestEntry(first)) { K key = first.key; removeNode(hash(key), key, null, false, true); } } protected boolean removeEldestEntry(Map.Entry eldest) { return false; }
細(xì)心的你也花現(xiàn)了吧。afterNodeInsertion()由于removeEldestEntry()所返回的false無(wú)執(zhí)行意義。也就意味著如果想要讓它有意義必須重寫(xiě)removeEldestEntry()。
如,使用LinkedHashMap實(shí)現(xiàn)一個(gè)簡(jiǎn)單的LRU(Least Recently Used)Cache。那么就應(yīng)該重寫(xiě)removeEldestEntry(),當(dāng)超出緩存容器大小時(shí)移除最老的首節(jié)點(diǎn)(這里不考慮并發(fā)問(wèn)題,如下):
@Override public boolean removeEldestEntry(Map.Entry? afterNodeRemoval()eldest){ return size() > capacity; }
void afterNodeRemoval(Nodee) { // unlink LinkedHashMap.Entry p = (LinkedHashMap.Entry )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; }
afterNodeRemoval()方法相對(duì)簡(jiǎn)單,就是在刪除后處理其對(duì)應(yīng)鏈表前后關(guān)系(刨掉一截)。
? internalWriteEntries()LinkedHashMap源碼閱讀總體門(mén)檻相對(duì)而言比HashMap,畢竟大多數(shù)底層put,get都由HashMap實(shí)現(xiàn)了。internalWriteEntries()相對(duì)來(lái)說(shuō)比較突兀,如果你知道它在哪里起著什么樣神秘的作用請(qǐng)?jiān)谠u(píng)論里告訴在下吧。[比心?]
void internalWriteEntries(java.io.ObjectOutputStream s) throws IOException { for (LinkedHashMap.Entrye = head; e != null; e = e.after) { s.writeObject(e.key); s.writeObject(e.value); } }
可通過(guò)這篇文章理解創(chuàng)建一個(gè)LinkedHashMap實(shí)例過(guò)程(圖):
Java_LinkedHashMap工作原理 2017-05-04;
往期線(xiàn)路回顧:
【java源碼一帶一路系列】之HashMap.putAll()
【java源碼一帶一路系列】之HashMap.putVal()
【java源碼一帶一路系列】之HashMap.compute()
文章版權(quán)歸作者所有,未經(jīng)允許請(qǐng)勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。
轉(zhuǎn)載請(qǐng)注明本文地址:http://specialneedsforspecialkids.com/yun/70089.html
摘要:一路至此,風(fēng)景過(guò)半。與雖然名字各異,源碼實(shí)現(xiàn)基本相同,除了增加了線(xiàn)程安全。同時(shí)注意溢出情況處理。同時(shí)增加了考慮并發(fā)問(wèn)題。此外,源碼中出現(xiàn)了大量泛型如。允許為非線(xiàn)程安全有序。 一路至此,風(fēng)景過(guò)半。ArrayList與Vector雖然名字各異,源碼實(shí)現(xiàn)基本相同,除了Vector增加了線(xiàn)程安全。所以作者建議我們?cè)诓恍枰€(xiàn)程安全的情況下盡量使用ArrayList。下面看看在ArrayList源...
摘要:同樣在源碼的與分別見(jiàn)看到老朋友和。這樣做可以降低性能消耗的同時(shí),還可以減少序列化字節(jié)流的大小,從而減少網(wǎng)絡(luò)開(kāi)銷(xiāo)框架中。使用了反射來(lái)尋找是否聲明了這兩個(gè)方法。十進(jìn)制和,通過(guò)返回值能反應(yīng)當(dāng)前狀態(tài)。 Map篇暫告段落,卻并非離我們而去。這不在本篇中你就能經(jīng)常見(jiàn)到她。HashSet、LinkedHashSet、TreeSet各自基于對(duì)應(yīng)Map實(shí)現(xiàn),各自源碼內(nèi)容較少,因此歸納為一篇。 HashS...
摘要:本篇涉及少許以下簡(jiǎn)稱(chēng)新特性,請(qǐng)?bào)H友們系好安全帶,準(zhǔn)備開(kāi)車(chē)。觀光線(xiàn)路圖是在中新增的一個(gè)方法,相對(duì)而言較為陌生。其作用是把的計(jì)算結(jié)果關(guān)聯(lián)到上即返回值作為新。實(shí)際上,乃縮寫(xiě),即二元函數(shù)之意類(lèi)似。 本文以jdk1.8中HashMap.compute()方法為切入點(diǎn),分析其中難理解、有價(jià)值的源碼片段(類(lèi)似源碼查看是ctrl+鼠標(biāo)左鍵的過(guò)程)。本篇涉及少許Java8(以下簡(jiǎn)稱(chēng)J8)新特性,請(qǐng)?bào)H友們...
摘要:觀光線(xiàn)路圖將涉及到的源碼全局變量哈希表初始化長(zhǎng)度默認(rèn)值是負(fù)載因子默認(rèn)表示的填滿(mǎn)程度。根據(jù)是否為零將原鏈表拆分成個(gè)鏈表,一部分仍保留在原鏈表中不需要移動(dòng),一部分移動(dòng)到原索引的新鏈表中。 前言 本文以jdk1.8中HashMap.putAll()方法為切入點(diǎn),分析其中難理解、有價(jià)值的源碼片段(類(lèi)似ctrl+鼠標(biāo)左鍵查看的源碼過(guò)程)。?觀光線(xiàn)路圖:putAll() --> putMapEnt...
摘要:表示該類(lèi)本身不可比表示與對(duì)應(yīng)的之間不可比。當(dāng)數(shù)目滿(mǎn)足時(shí),鏈表將轉(zhuǎn)為紅黑樹(shù)結(jié)構(gòu),否則繼續(xù)擴(kuò)容。至此,插入告一段落。當(dāng)超出時(shí),哈希表將會(huì)即內(nèi)部數(shù)據(jù)結(jié)構(gòu)重建至大約兩倍。要注意的是使用許多有這相同的鍵值肯定會(huì)降低哈希表性能。 回顧上期?觀光線(xiàn)路圖:putAll() --> putMapEntries() --> tableSizeFor() --> resize() --> hash() --...
閱讀 3026·2021-11-24 09:39
閱讀 2255·2021-10-08 10:05
閱讀 2749·2021-09-24 13:52
閱讀 1569·2021-09-22 15:07
閱讀 589·2019-08-30 15:55
閱讀 1808·2019-08-30 15:53
閱讀 687·2019-08-30 15:44
閱讀 3116·2019-08-30 11:20