日本不卡不码高清免费观看,久久国产精品久久w女人spa,黄色aa久久,三上悠亚国产精品一区二区三区

您的位置:首頁技術文章
文章詳情頁

Java源碼解析之ConcurrentHashMap

瀏覽:171日期:2022-08-11 17:31:33

早期 ConcurrentHashMap,其實現是基于:

分離鎖,也就是將內部進行分段(Segment),里面則是 HashEntry 的數組,和 HashMap 類似,哈希相同的條目也是以鏈表形式存放。 HashEntry 內部使用 volatile 的 value 字段來保證可見性,也利用了不可變對象的機制以改進利用 Unsafe 提供的底層能力,比如 volatile access,去直接完成部分操作,以最優化性能,畢竟 Unsafe 中的很多操作都是 JVM intrinsic 優化過的。

在進行并發操作的時候,只需要鎖定相應段,這樣就有效避免了類似 Hashtable 整體同步的問題,大大提高了性能。

Java源碼解析之ConcurrentHashMap

Put操作

通過二次哈希避免哈希沖突,然后以 Unsafe 調用方式,直接獲取相應的 Segment,然后進行線程安全的 put 操作

public V put(K key, V value) { Segment<K,V> s; if (value == null) throw new NullPointerException(); // 二次哈希,以保證數據的分散性,避免哈希沖突 int hash = hash(key.hashCode()); int j = (hash >>> segmentShift) & segmentMask; if ((s = (Segment<K,V>)UNSAFE.getObject // nonvolatile; recheck (segments, (j << SSHIFT) + SBASE)) == null) // in ensureSegment s = ensureSegment(j); return s.put(key, hash, value, false); }

其核心邏輯實現在下面的內部方法中:

final V put(K key, int hash, V value, boolean onlyIfAbsent) { // scanAndLockForPut 會去查找是否有 key 相同 Node // 無論如何,確保獲取鎖 HashEntry<K,V> node = tryLock() ? null : scanAndLockForPut(key, hash, value); V oldValue; try { HashEntry<K,V>[] tab = table; int index = (tab.length - 1) & hash; HashEntry<K,V> first = entryAt(tab, index); for (HashEntry<K,V> e = first;;) { if (e != null) { K k; // 更新已有 value... } else { // 放置 HashEntry 到特定位置,如果超過閾值,進行 rehash // ... } } } finally { unlock(); } return oldValue; }

在寫的時候:

ConcurrentHashMap 會獲取再入鎖,以保證數據一致性,Segment 本身就是基于 ReentrantLock 的擴展實現,所以,在并發修改期間,相應 Segment 是被鎖定的。 在最初階段,進行重復性的掃描,以確定相應 key 值是否已經在數組里面,進而決定是更新還是放置操作。 在 ConcurrentHashMap 中解決擴容的問題,不是整體的擴容,而是單獨對 Segment 進行擴容。 為了減少鎖定segment的開銷,ConcurrentHashMap 的實現是通過重試機制(RETRIES_BEFORE_LOCK,指定重試次數 2),來試圖獲得可靠值。如果沒有監控到發生變化(通過對比 Segment.modCount),就直接返回,否則獲取鎖進行操作。

機制在Java 8 上的變化:

總體結構上,它的內部存儲與HashMap 結構非常相似,同樣是大的桶(bucket)數組,然后內部也是一個個所謂的鏈表結構(bin),同步的粒度要更細致一些。 其內部仍然有 Segment 定義,但僅僅是為了保證序列化時的兼容性而已,不再有任何結構上的用處。 因為不再使用 Segment,初始化操作大大簡化,修改為 lazy-load 形式,這樣可以有效避免初始開銷。 數據存儲利用 volatile 來保證可見性。 使用 CAS (Compare And Swap)等操作,在特定場景進行無鎖并發操作。 使用 Unsafe、LongAdder 之類底層手段,進行極端情況的優化。

看看在java8上的put操作

final V putVal(K key, V value, boolean onlyIfAbsent) { if (key == null || value == null) throw new NullPointerException(); int hash = spread(key.hashCode()); int binCount = 0; for (Node<K,V>[] tab = table;;) { Node<K,V> f; int n, i, fh; K fk; V fv; if (tab == null || (n = tab.length) == 0) tab = initTable(); else if ((f = tabAt(tab, i = (n - 1) & hash)) == null) { // 利用 CAS 去進行無鎖線程安全操作,如果 bin 是空的 if (casTabAt(tab, i, null, new Node<K,V>(hash, key, value))) break; } else if ((fh = f.hash) == MOVED) tab = helpTransfer(tab, f); else if (onlyIfAbsent // 不加鎖,進行檢查 && fh == hash && ((fk = f.key) == key || (fk != null && key.equals(fk))) && (fv = f.val) != null) return fv; else { V oldVal = null; synchronized (f) { // 細粒度的同步修改操作... } } // Bin 超過閾值,進行樹化 if (binCount != 0) { if (binCount >= TREEIFY_THRESHOLD) treeifyBin(tab, i); if (oldVal != null) return oldVal; break; } } } addCount(1L, binCount); return null; }

初始化操作實現在 initTable 里面,這是一個典型的 CAS 使用場景,利用 volatile 的 sizeCtl 作為互斥手段:如果發現競爭性的初始化,就 spin 在那里,等待條件恢復;否則利用 CAS 設置排他標志。如果成功則進行初始化;否則重試。

private final Node<K,V>[] initTable() { Node<K,V>[] tab; int sc; while ((tab = table) == null || tab.length == 0) { // 如果發現沖突,進行 spin 等待 if ((sc = sizeCtl) < 0) Thread.yield(); // CAS 成功返回 true,則進入真正的初始化邏輯 else if (U.compareAndSetInt(this, SIZECTL, sc, -1)) { try { if ((tab = table) == null || tab.length == 0) { int n = (sc > 0) ? sc : DEFAULT_CAPACITY; @SuppressWarnings('unchecked') Node<K,V>[] nt = (Node<K,V>[])new Node<?,?>[n]; table = tab = nt; sc = n - (n >>> 2); } } finally { sizeCtl = sc; } break; } } return tab; }

當 bin 為空時,同樣是沒有必要鎖定,也是以 CAS 操作去放置。

到此這篇關于Java源碼解析之ConcurrentHashMap的文章就介紹到這了,更多相關Java ConcurrentHashMap內容請搜索好吧啦網以前的文章或繼續瀏覽下面的相關文章希望大家以后多多支持好吧啦網!

標簽: Java
相關文章:
日本不卡不码高清免费观看,久久国产精品久久w女人spa,黄色aa久久,三上悠亚国产精品一区二区三区
国产精品日韩欧美一区| 国产亚洲欧洲| 国产三级精品三级在线观看国产| 在线看片日韩| 亚洲一二三区视频| 日本一区二区三区中文字幕| 亚洲一区av| 日本伊人久久| 日韩高清二区| 欧美一区成人| 麻豆视频久久| 成人一区而且| 美女视频免费精品| 国产专区精品| 日韩高清成人| 久久精品国产99久久| 婷婷六月综合| 免费精品视频最新在线| 五月亚洲婷婷| 国产欧美88| 狂野欧美性猛交xxxx| 美女视频免费精品| 麻豆理论在线观看| 激情综合网五月| 视频一区视频二区中文| 日韩在线电影| 国产一区二区三区黄网站 | 一区二区精品伦理...| 欧美成a人国产精品高清乱码在线观看片在线观看久 | 精品理论电影在线| 国产精品原创| 欧美精品一卡| 日韩精品久久久久久| 日韩精品免费一区二区夜夜嗨| 国产亚洲久久| 国产福利片在线观看| 免费成人网www| 日韩午夜视频在线| 麻豆中文一区二区| 国产伊人精品| 日韩激情视频网站| 日韩国产综合| 日韩精品一区第一页| 嫩呦国产一区二区三区av| 97精品中文字幕| 日韩亚洲国产欧美| 69堂精品视频在线播放| 中文字幕在线看片| 99在线观看免费视频精品观看| 综合一区av| 精品亚洲自拍| 精品一区毛片| 青青草国产精品亚洲专区无| 97精品一区二区| 国产精品婷婷| 久久久天天操| 日韩国产一区二| 国产一区二区三区不卡av| 亚洲午夜一级| 国产伦精品一区二区三区视频| 日韩理论片av| 日韩精品亚洲专区| 亚洲风情在线资源| 一区二区三区四区日韩| 精品国产91| 免费人成在线不卡| 福利一区在线| 美女网站一区| 国产精品亚洲产品| 国产视频亚洲| 久久伊人亚洲| 国产视频欧美| 麻豆精品99| 天堂va蜜桃一区二区三区| 成人在线丰满少妇av| 青草久久视频| 狠狠久久婷婷| 久久99精品久久久野外观看| 精品1区2区3区4区| 精品免费av| 青草av.久久免费一区| 欧美 日韩 国产一区二区在线视频| 国产伦精品一区二区三区在线播放| 婷婷成人基地| 国产美女高潮在线观看| 欧美啪啪一区| 男人操女人的视频在线观看欧美| 免费视频一区二区三区在线观看 | 蜜臀精品一区二区三区在线观看 | 国产亚洲字幕| 亚洲在线成人| 久久免费大视频| 国产自产自拍视频在线观看| 国产精品1区在线| 丝袜a∨在线一区二区三区不卡| 91亚洲自偷观看高清| 国产精品手机在线播放| 亚洲精品无吗| 久热精品在线| 中日韩男男gay无套| 久久中文视频| 日韩一区欧美| 成人日韩av| 免费在线成人| 国产精品中文| 欧美日韩一区二区国产| 日韩精品亚洲aⅴ在线影院| 免费日韩av片| 日韩视频一区| 久久精品青草| 日韩免费久久| 亚洲黄色网址| 四虎成人av| 国产在线观看91一区二区三区| 欧美亚洲色图校园春色| 日韩欧美三区| 日韩精品一级| 欧美在线看片| 国产日韩精品视频一区二区三区| 日韩国产精品久久久久久亚洲| 亚洲乱码一区| 日韩精品一区二区三区中文在线| 亚洲精品激情| 涩涩涩久久久成人精品| 首页国产欧美久久| 亚洲欧美网站| 蜜臀91精品一区二区三区| 老鸭窝毛片一区二区三区| 欧美日韩国产精品一区二区亚洲| 亚洲国产一区二区在线观看| 91九色精品| 久久国产精品久久w女人spa| 首页国产欧美日韩丝袜| 亚洲小说春色综合另类电影| 午夜久久av| 国产日韩一区| 成人精品高清在线视频| 国产精品yjizz视频网| 久久精品国语| 亚洲欧洲一区二区天堂久久| 欧美色图一区| 午夜精品影院| 中文字幕日韩亚洲| 88久久精品| 九九久久国产| 六月婷婷综合| 激情久久五月| 亚洲作爱视频| 午夜电影一区| 欧美黑人巨大videos精品| 国产精品成人a在线观看| 日韩在线短视频| 蜜桃国内精品久久久久软件9| 99在线|亚洲一区二区| 另类av一区二区| 亚洲另类黄色| 国产精品视频一区二区三区综合| 精品美女视频| 激情婷婷综合| 日韩一区二区三区高清在线观看| 日本欧美一区二区在线观看| 国产精品羞羞答答在线观看| 国产一区二区三区91| 国产99精品一区| 亚洲精品美女91| 嫩草伊人久久精品少妇av杨幂 | 麻豆理论在线观看| 99pao成人国产永久免费视频 | 在线视频观看日韩| 亚洲香蕉视频| 精品一区二区三区亚洲| 欧美精品九九| 久久国产欧美日韩精品| 国产美女高潮在线| 国产精品人人爽人人做我的可爱 | 亚洲人妖在线| 精品国产午夜| 欧美在线亚洲综合一区| 亚洲三级在线| 国产在线不卡一区二区三区| 婷婷激情图片久久| 国产精品欧美大片| 亚洲精品2区| 久久99偷拍| 伊人成人网在线看| 国产精品xxxav免费视频| 美女网站一区| 久久不卡国产精品一区二区| 女人av一区| 麻豆国产精品一区二区三区| 亚洲激情二区| 另类欧美日韩国产在线| 午夜久久99| 欧美永久精品| 欧美.日韩.国产.一区.二区| 久久精品99久久久| 午夜欧美在线| 粉嫩av一区二区三区四区五区 | 欧洲激情综合| 国产精品毛片久久久|