日韩性视频-久久久蜜桃-www中文字幕-在线中文字幕av-亚洲欧美一区二区三区四区-撸久久-香蕉视频一区-久久无码精品丰满人妻-国产高潮av-激情福利社-日韩av网址大全-国产精品久久999-日本五十路在线-性欧美在线-久久99精品波多结衣一区-男女午夜免费视频-黑人极品ⅴideos精品欧美棵-人人妻人人澡人人爽精品欧美一区-日韩一区在线看-欧美a级在线免费观看

歡迎訪問 生活随笔!

生活随笔

當(dāng)前位置: 首頁 > 编程资源 > 编程问答 >内容正文

编程问答

ConcurrentHashMap之实现细节(转)

發(fā)布時間:2024/4/15 编程问答 45 豆豆
生活随笔 收集整理的這篇文章主要介紹了 ConcurrentHashMap之实现细节(转) 小編覺得挺不錯的,現(xiàn)在分享給大家,幫大家做個參考.

ConcurrentHashMap是Java 5中支持高并發(fā)、高吞吐量的線程安全HashMap實現(xiàn)。在這之前我對ConcurrentHashMap只有一些膚淺的理解,僅知道它采用了多個鎖,大概也足夠了。但是在經(jīng)過一次慘痛的面試經(jīng)歷之后,我覺得必須深入研究它的實現(xiàn)。面試中被問到讀是否要加鎖,因為讀寫會發(fā)生沖突,我說必須要加鎖,我和面試官也因此發(fā)生了沖突,結(jié)果可想而知。還是閑話少說,通過仔細閱讀源代碼,現(xiàn)在總算理解ConcurrentHashMap實現(xiàn)機制了,其實現(xiàn)之精巧,令人嘆服,與大家共享之。

?

?

實現(xiàn)原理?

?

鎖分離 (Lock Stripping)

?

ConcurrentHashMap允許多個修改操作并發(fā)進行,其關(guān)鍵在于使用了鎖分離技術(shù)。它使用了多個鎖來控制對hash表的不同部分進行的修改。ConcurrentHashMap內(nèi)部使用段(Segment)來表示這些不同的部分,每個段其實就是一個小的hash table,它們有自己的鎖。只要多個修改操作發(fā)生在不同的段上,它們就可以并發(fā)進行。

?

有些方法需要跨段,比如size()和containsValue(),它們可能需要鎖定整個表而而不僅僅是某個段,這需要按順序鎖定所有段,操作完畢后,又按順序釋放所有段的鎖。這里“按順序”是很重要的,否則極有可能出現(xiàn)死鎖,在ConcurrentHashMap內(nèi)部,段數(shù)組是final的,并且其成員變量實際上也是final的,但是,僅僅是將數(shù)組聲明為final的并不保證數(shù)組成員也是final的,這需要實現(xiàn)上的保證。這可以確保不會出現(xiàn)死鎖,因為獲得鎖的順序是固定的。不變性是多線程編程占有很重要的地位,下面還要談到。

?

Java代碼??
  • /**?
  • ?*?The?segments,?each?of?which?is?a?specialized?hash?table?
  • ?*/??
  • final?Segment<K,V>[]?segments;??
  • ?

    ?

    不變(Immutable)和易變(Volatile)

    ?

    ConcurrentHashMap完全允許多個讀操作并發(fā)進行,讀操作并不需要加鎖。如果使用傳統(tǒng)的技術(shù),如HashMap中的實現(xiàn),如果允許可以在hash鏈的中間添加或刪除元素,讀操作不加鎖將得到不一致的數(shù)據(jù)。ConcurrentHashMap實現(xiàn)技術(shù)是保證HashEntry幾乎是不可變的。HashEntry代表每個hash鏈中的一個節(jié)點,其結(jié)構(gòu)如下所示:

    ?

    Java代碼??
  • static?final?class?HashEntry<K,V>?{??
  • ????final?K?key;??
  • ????final?int?hash;??
  • ????volatile?V?value;??
  • ????final?HashEntry<K,V>?next;??
  • }??
  • 可以看到除了value不是final的,其它值都是final的,這意味著不能從hash鏈的中間或尾部添加或刪除節(jié)點,因為這需要修改next引用值,所有的節(jié)點的修改只能從頭部開始。對于put操作,可以一律添加到Hash鏈的頭部。但是對于remove操作,可能需要從中間刪除一個節(jié)點,這就需要將要刪除節(jié)點的前面所有節(jié)點整個復(fù)制一遍,最后一個節(jié)點指向要刪除結(jié)點的下一個結(jié)點。這在講解刪除操作時還會詳述。為了確保讀操作能夠看到最新的值,將value設(shè)置成volatile,這避免了加鎖。

    ?


    其它

    ?

    為了加快定位段以及段中hash槽的速度,每個段hash槽的的個數(shù)都是2^n,這使得通過位運算就可以定位段和段中hash槽的位置。當(dāng)并發(fā)級別為默認(rèn)值16時,也就是段的個數(shù),hash值的高4位決定分配在哪個段中。但是我們也不要忘記《算法導(dǎo)論》給我們的教訓(xùn):hash槽的的個數(shù)不應(yīng)該是2^n,這可能導(dǎo)致hash槽分配不均,這需要對hash值重新再hash一次。(這段似乎有點多余了?)

    ?

    這是重新hash的算法,還比較復(fù)雜,我也懶得去理解了。

    Java代碼??
  • private?static?int?hash(int?h)?{??
  • ????//?Spread?bits?to?regularize?both?segment?and?index?locations,??
  • ????//?using?variant?of?single-word?Wang/Jenkins?hash.??
  • ????h?+=?(h?<<??15)?^?0xffffcd7d;??
  • ????h?^=?(h?>>>?10);??
  • ????h?+=?(h?<<???3);??
  • ????h?^=?(h?>>>??6);??
  • ????h?+=?(h?<<???2)?+?(h?<<?14);??
  • ????return?h?^?(h?>>>?16);??
  • }??
  • ?

    這是定位段的方法:

    Java代碼??
  • final?Segment<K,V>?segmentFor(int?hash)?{??
  • ????return?segments[(hash?>>>?segmentShift)?&?segmentMask];??
  • }??
  • ?

    ?

    ?

    數(shù)據(jù)結(jié)構(gòu)

    ?

    關(guān)于Hash表的基礎(chǔ)數(shù)據(jù)結(jié)構(gòu),這里不想做過多的探討。Hash表的一個很重要方面就是如何解決hash沖突,ConcurrentHashMap和HashMap使用相同的方式,都是將hash值相同的節(jié)點放在一個hash鏈中。與HashMap不同的是,ConcurrentHashMap使用多個子Hash表,也就是段(Segment)。下面是ConcurrentHashMap的數(shù)據(jù)成員:

    ?

    Java代碼??
  • public?class?ConcurrentHashMap<K,?V>?extends?AbstractMap<K,?V>??
  • ????????implements?ConcurrentMap<K,?V>,?Serializable?{??
  • ????/**?
  • ?????*?Mask?value?for?indexing?into?segments.?The?upper?bits?of?a?
  • ?????*?key's?hash?code?are?used?to?choose?the?segment.?
  • ?????*/??
  • ????final?int?segmentMask;??
  • ??
  • ????/**?
  • ?????*?Shift?value?for?indexing?within?segments.?
  • ?????*/??
  • ????final?int?segmentShift;??
  • ??
  • ????/**?
  • ?????*?The?segments,?each?of?which?is?a?specialized?hash?table?
  • ?????*/??
  • ????final?Segment<K,V>[]?segments;??
  • }??
  • ?

    所有的成員都是final的,其中segmentMask和segmentShift主要是為了定位段,參見上面的segmentFor方法。

    ?

    每個Segment相當(dāng)于一個子Hash表,它的數(shù)據(jù)成員如下:

    ?

    Java代碼??
  • ????static?final?class?Segment<K,V>?extends?ReentrantLock?implements?Serializable?{??
  • private?static?final?long?serialVersionUID?=?2249069246763182397L;??
  • ????????/**?
  • ?????????*?The?number?of?elements?in?this?segment's?region.?
  • ?????????*/??
  • ????????transient?volatile?int?count;??
  • ??
  • ????????/**?
  • ?????????*?Number?of?updates?that?alter?the?size?of?the?table.?This?is?
  • ?????????*?used?during?bulk-read?methods?to?make?sure?they?see?a?
  • ?????????*?consistent?snapshot:?If?modCounts?change?during?a?traversal?
  • ?????????*?of?segments?computing?size?or?checking?containsValue,?then?
  • ?????????*?we?might?have?an?inconsistent?view?of?state?so?(usually)?
  • ?????????*?must?retry.?
  • ?????????*/??
  • ????????transient?int?modCount;??
  • ??
  • ????????/**?
  • ?????????*?The?table?is?rehashed?when?its?size?exceeds?this?threshold.?
  • ?????????*?(The?value?of?this?field?is?always?<tt>(int)(capacity?*?
  • ?????????*?loadFactor)</tt>.)?
  • ?????????*/??
  • ????????transient?int?threshold;??
  • ??
  • ????????/**?
  • ?????????*?The?per-segment?table.?
  • ?????????*/??
  • ????????transient?volatile?HashEntry<K,V>[]?table;??
  • ??
  • ????????/**?
  • ?????????*?The?load?factor?for?the?hash?table.??Even?though?this?value?
  • ?????????*?is?same?for?all?segments,?it?is?replicated?to?avoid?needing?
  • ?????????*?links?to?outer?object.?
  • ?????????*?@serial?
  • ?????????*/??
  • ????????final?float?loadFactor;??
  • }??
  • count用來統(tǒng)計該段數(shù)據(jù)的個數(shù),它是volatile,它用來協(xié)調(diào)修改和讀取操作,以保證讀取操作能夠讀取到幾乎最新的修改。協(xié)調(diào)方式是這樣的,每次修改操作做了結(jié)構(gòu)上的改變,如增加/刪除節(jié)點(修改節(jié)點的值不算結(jié)構(gòu)上的改變),都要寫count值,每次讀取操作開始都要讀取count的值。這利用了Java 5中對volatile語義的增強,對同一個volatile變量的寫和讀存在happens-before關(guān)系。modCount統(tǒng)計段結(jié)構(gòu)改變的次數(shù),主要是為了檢測對多個段進行遍歷過程中某個段是否發(fā)生改變,在講述跨段操作時會還會詳述。threashold用來表示需要進行rehash的界限值。table數(shù)組存儲段中節(jié)點,每個數(shù)組元素是個hash鏈,用HashEntry表示。table也是volatile,這使得能夠讀取到最新的table值而不需要同步。loadFactor表示負載因子。

    ?

    ?

    實現(xiàn)細節(jié)

    ?

    修改操作

    ?

    先來看下刪除操作remove(key)。

    Java代碼??
  • public?V?remove(Object?key)?{??
  • ?hash?=?hash(key.hashCode());??
  • ????return?segmentFor(hash).remove(key,?hash,?null);??
  • }??
  • 整個操作是先定位到段,然后委托給段的remove操作。當(dāng)多個刪除操作并發(fā)進行時,只要它們所在的段不相同,它們就可以同時進行。下面是Segment的remove方法實現(xiàn):

    Java代碼??
  • V?remove(Object?key,?int?hash,?Object?value)?{??
  • ????lock();??
  • ????try?{??
  • ????????int?c?=?count?-?1;??
  • ????????HashEntry<K,V>[]?tab?=?table;??
  • ????????int?index?=?hash?&?(tab.length?-?1);??
  • ????????HashEntry<K,V>?first?=?tab[index];??
  • ????????HashEntry<K,V>?e?=?first;??
  • ????????while?(e?!=?null?&&?(e.hash?!=?hash?||?!key.equals(e.key)))??
  • ????????????e?=?e.next;??
  • ??
  • ????????V?oldValue?=?null;??
  • ????????if?(e?!=?null)?{??
  • ????????????V?v?=?e.value;??
  • ????????????if?(value?==?null?||?value.equals(v))?{??
  • ????????????????oldValue?=?v;??
  • ????????????????//?All?entries?following?removed?node?can?stay??
  • ????????????????//?in?list,?but?all?preceding?ones?need?to?be??
  • ????????????????//?cloned.??
  • ????????????????++modCount;??
  • ????????????????HashEntry<K,V>?newFirst?=?e.next;??
  • ????????????????for?(HashEntry<K,V>?p?=?first;?p?!=?e;?p?=?p.next)??
  • ????????????????????newFirst?=?new?HashEntry<K,V>(p.key,?p.hash,??
  • ??????????????????????????????????????????????????newFirst,?p.value);??
  • ????????????????tab[index]?=?newFirst;??
  • ????????????????count?=?c;?//?write-volatile??
  • ????????????}??
  • ????????}??
  • ????????return?oldValue;??
  • ????}?finally?{??
  • ????????unlock();??
  • ????}??
  • }??
  • ?整個操作是在持有段鎖的情況下執(zhí)行的,空白行之前的行主要是定位到要刪除的節(jié)點e。接下來,如果不存在這個節(jié)點就直接返回null,否則就要將e前面的結(jié)點復(fù)制一遍,尾結(jié)點指向e的下一個結(jié)點。e后面的結(jié)點不需要復(fù)制,它們可以重用。下面是個示意圖,我直接從這個網(wǎng)站?上復(fù)制的(畫這樣的圖實在是太麻煩了,如果哪位有好的畫圖工具,可以推薦一下)。

    ?

    ?

    刪除元素之前:

    ?

    ?

    ?

    刪除元素3之后:

    ?

    第二個圖其實有點問題,復(fù)制的結(jié)點中應(yīng)該是值為2的結(jié)點在前面,值為1的結(jié)點在后面,也就是剛好和原來結(jié)點順序相反,還好這不影響我們的討論。

    ?

    整個remove實現(xiàn)并不復(fù)雜,但是需要注意如下幾點。第一,當(dāng)要刪除的結(jié)點存在時,刪除的最后一步操作要將count的值減一。這必須是最后一步操作,否則讀取操作可能看不到之前對段所做的結(jié)構(gòu)性修改。第二,remove執(zhí)行的開始就將table賦給一個局部變量tab,這是因為table是volatile變量,讀寫volatile變量的開銷很大。編譯器也不能對volatile變量的讀寫做任何優(yōu)化,直接多次訪問非volatile實例變量沒有多大影響,編譯器會做相應(yīng)優(yōu)化。

    ?

    ?

    接下來看put操作,同樣地put操作也是委托給段的put方法。下面是段的put方法:

    Java代碼??
  • V?put(K?key,?int?hash,?V?value,?boolean?onlyIfAbsent)?{??
  • ????lock();??
  • ????try?{??
  • ????????int?c?=?count;??
  • ????????if?(c++?>?threshold)?//?ensure?capacity??
  • ????????????rehash();??
  • ????????HashEntry<K,V>[]?tab?=?table;??
  • ????????int?index?=?hash?&?(tab.length?-?1);??
  • ????????HashEntry<K,V>?first?=?tab[index];??
  • ????????HashEntry<K,V>?e?=?first;??
  • ????????while?(e?!=?null?&&?(e.hash?!=?hash?||?!key.equals(e.key)))??
  • ????????????e?=?e.next;??
  • ??
  • ????????V?oldValue;??
  • ????????if?(e?!=?null)?{??
  • ????????????oldValue?=?e.value;??
  • ????????????if?(!onlyIfAbsent)??
  • ????????????????e.value?=?value;??
  • ????????}??
  • ????????else?{??
  • ????????????oldValue?=?null;??
  • ????????????++modCount;??
  • ????????????tab[index]?=?new?HashEntry<K,V>(key,?hash,?first,?value);??
  • ????????????count?=?c;?//?write-volatile??
  • ????????}??
  • ????????return?oldValue;??
  • ????}?finally?{??
  • ????????unlock();??
  • ????}??
  • }??
  • 該方法也是在持有段鎖的情況下執(zhí)行的,首先判斷是否需要rehash,需要就先rehash。接著是找是否存在同樣一個key的結(jié)點,如果存在就直接替換這個結(jié)點的值。否則創(chuàng)建一個新的結(jié)點并添加到hash鏈的頭部,這時一定要修改modCount和count的值,同樣修改count的值一定要放在最后一步。put方法調(diào)用了rehash方法,reash方法實現(xiàn)得也很精巧,主要利用了table的大小為2^n,這里就不介紹了。

    ?

    修改操作還有putAll和replace。putAll就是多次調(diào)用put方法,沒什么好說的。replace甚至不用做結(jié)構(gòu)上的更改,實現(xiàn)要比put和delete要簡單得多,理解了put和delete,理解replace就不在話下了,這里也不介紹了。

    ?

    ?

    獲取操作

    ?

    首先看下get操作,同樣ConcurrentHashMap的get操作是直接委托給Segment的get方法,直接看Segment的get方法:

    ?

    Java代碼??
  • V?get(Object?key,?int?hash)?{??
  • ????if?(count?!=?0)?{?//?read-volatile??
  • ????????HashEntry<K,V>?e?=?getFirst(hash);??
  • ????????while?(e?!=?null)?{??
  • ????????????if?(e.hash?==?hash?&&?key.equals(e.key))?{??
  • ????????????????V?v?=?e.value;??
  • ????????????????if?(v?!=?null)??
  • ????????????????????return?v;??
  • ????????????????return?readValueUnderLock(e);?//?recheck??
  • ????????????}??
  • ????????????e?=?e.next;??
  • ????????}??
  • ????}??
  • ????return?null;??
  • }??
  • ?

    get操作不需要鎖。第一步是訪問count變量,這是一個volatile變量,由于所有的修改操作在進行結(jié)構(gòu)修改時都會在最后一步寫count變量,通過這種機制保證get操作能夠得到幾乎最新的結(jié)構(gòu)更新。對于非結(jié)構(gòu)更新,也就是結(jié)點值的改變,由于HashEntry的value變量是volatile的,也能保證讀取到最新的值。接下來就是對hash鏈進行遍歷找到要獲取的結(jié)點,如果沒有找到,直接訪回null。對hash鏈進行遍歷不需要加鎖的原因在于鏈指針next是final的。但是頭指針卻不是final的,這是通過getFirst(hash)方法返回,也就是存在table數(shù)組中的值。這使得getFirst(hash)可能返回過時的頭結(jié)點,例如,當(dāng)執(zhí)行g(shù)et方法時,剛執(zhí)行完getFirst(hash)之后,另一個線程執(zhí)行了刪除操作并更新頭結(jié)點,這就導(dǎo)致get方法中返回的頭結(jié)點不是最新的。這是可以允許,通過對count變量的協(xié)調(diào)機制,get能讀取到幾乎最新的數(shù)據(jù),雖然可能不是最新的。要得到最新的數(shù)據(jù),只有采用完全的同步。

    ?

    最后,如果找到了所求的結(jié)點,判斷它的值如果非空就直接返回,否則在有鎖的狀態(tài)下再讀一次。這似乎有些費解,理論上結(jié)點的值不可能為空,這是因為put的時候就進行了判斷,如果為空就要拋NullPointerException。空值的唯一源頭就是HashEntry中的默認(rèn)值,因為HashEntry中的value不是final的,非同步讀取有可能讀取到空值。仔細看下put操作的語句:tab[index] = new HashEntry<K,V>(key, hash, first, value),在這條語句中,HashEntry構(gòu)造函數(shù)中對value的賦值以及對tab[index]的賦值可能被重新排序,這就可能導(dǎo)致結(jié)點的值為空。這種情況應(yīng)當(dāng)很罕見,一旦發(fā)生這種情況,ConcurrentHashMap采取的方式是在持有鎖的情況下再讀一遍,這能夠保證讀到最新的值,并且一定不會為空值。

    ?

    Java代碼??
  • V?readValueUnderLock(HashEntry<K,V>?e)?{??
  • ????lock();??
  • ????try?{??
  • ????????return?e.value;??
  • ????}?finally?{??
  • ????????unlock();??
  • ????}??
  • }??
  • ?

    ?

    另一個操作是containsKey,這個實現(xiàn)就要簡單得多了,因為它不需要讀取值:

    Java代碼??
  • boolean?containsKey(Object?key,?int?hash)?{??
  • ????if?(count?!=?0)?{?//?read-volatile??
  • ????????HashEntry<K,V>?e?=?getFirst(hash);??
  • ????????while?(e?!=?null)?{??
  • ????????????if?(e.hash?==?hash?&&?key.equals(e.key))??
  • ????????????????return?true;??
  • ????????????e?=?e.next;??
  • ????????}??
  • ????}??
  • ????return?false;??
  • }??
  • ?

    ?

    跨段操作?

    ?

    有些操作需要涉及到多個段,比如說size(), containsValaue()。先來看下size()方法:

    ?

    Java代碼??
  • public?int?size()?{??
  • ????final?Segment<K,V>[]?segments?=?this.segments;??
  • ????long?sum?=?0;??
  • ????long?check?=?0;??
  • ????int[]?mc?=?new?int[segments.length];??
  • ????//?Try?a?few?times?to?get?accurate?count.?On?failure?due?to??
  • ????//?continuous?async?changes?in?table,?resort?to?locking.??
  • ????for?(int?k?=?0;?k?<?RETRIES_BEFORE_LOCK;?++k)?{??
  • ????????check?=?0;??
  • ????????sum?=?0;??
  • ????????int?mcsum?=?0;??
  • ????????for?(int?i?=?0;?i?<?segments.length;?++i)?{??
  • ????????????sum?+=?segments[i].count;??
  • ????????????mcsum?+=?mc[i]?=?segments[i].modCount;??
  • ????????}??
  • ????????if?(mcsum?!=?0)?{??
  • ????????????for?(int?i?=?0;?i?<?segments.length;?++i)?{??
  • ????????????????check?+=?segments[i].count;??
  • ????????????????if?(mc[i]?!=?segments[i].modCount)?{??
  • ????????????????????check?=?-1;?//?force?retry??
  • ????????????????????break;??
  • ????????????????}??
  • ????????????}??
  • ????????}??
  • ????????if?(check?==?sum)??
  • ????????????break;??
  • ????}??
  • ????if?(check?!=?sum)?{?//?Resort?to?locking?all?segments??
  • ????????sum?=?0;??
  • ????????for?(int?i?=?0;?i?<?segments.length;?++i)??
  • ????????????segments[i].lock();??
  • ????????for?(int?i?=?0;?i?<?segments.length;?++i)??
  • ????????????sum?+=?segments[i].count;??
  • ????????for?(int?i?=?0;?i?<?segments.length;?++i)??
  • ????????????segments[i].unlock();??
  • ????}??
  • ????if?(sum?>?Integer.MAX_VALUE)??
  • ????????return?Integer.MAX_VALUE;??
  • ????else??
  • ????????return?(int)sum;??
  • }??
  • ?

    size方法主要思路是先在沒有鎖的情況下對所有段大小求和,如果不能成功(這是因為遍歷過程中可能有其它線程正在對已經(jīng)遍歷過的段進行結(jié)構(gòu)性更新),最多執(zhí)行RETRIES_BEFORE_LOCK次,如果還不成功就在持有所有段鎖的情況下再對所有段大小求和。在沒有鎖的情況下主要是利用Segment中的modCount進行檢測,在遍歷過程中保存每個Segment的modCount,遍歷完成之后再檢測每個Segment的modCount有沒有改變,如果有改變表示有其它線程正在對Segment進行結(jié)構(gòu)性并發(fā)更新,需要重新計算。

    ?

    ?

    其實這種方式是存在問題的,在第一個內(nèi)層for循環(huán)中,在這兩條語句sum += segments[i].count; mcsum += mc[i] = segments[i].modCount;之間,其它線程可能正在對Segment進行結(jié)構(gòu)性的修改,導(dǎo)致segments[i].count和segments[i].modCount讀取的數(shù)據(jù)并不一致。這可能使size()方法返回任何時候都不曾存在的大小,很奇怪javadoc居然沒有明確標(biāo)出這一點,可能是因為這個時間窗口太小了吧。size()的實現(xiàn)還有一點需要注意,必須要先segments[i].count,才能segments[i].modCount,這是因為segment[i].count是對volatile變量的訪問,接下來segments[i].modCount才能得到幾乎最新的值(前面我已經(jīng)說了為什么只是“幾乎”了)。這點在containsValue方法中得到了淋漓盡致的展現(xiàn):

    ?

    ?

    Java代碼??
  • public?boolean?containsValue(Object?value)?{??
  • ????if?(value?==?null)??
  • ????????throw?new?NullPointerException();??
  • ??
  • ????//?See?explanation?of?modCount?use?above??
  • ??
  • ????final?Segment<K,V>[]?segments?=?this.segments;??
  • ????int[]?mc?=?new?int[segments.length];??
  • ??
  • ????//?Try?a?few?times?without?locking??
  • ????for?(int?k?=?0;?k?<?RETRIES_BEFORE_LOCK;?++k)?{??
  • ????????int?sum?=?0;??
  • ????????int?mcsum?=?0;??
  • ????????for?(int?i?=?0;?i?<?segments.length;?++i)?{??
  • ????????????int?c?=?segments[i].count;??
  • ????????????mcsum?+=?mc[i]?=?segments[i].modCount;??
  • ????????????if?(segments[i].containsValue(value))??
  • ????????????????return?true;??
  • ????????}??
  • ????????boolean?cleanSweep?=?true;??
  • ????????if?(mcsum?!=?0)?{??
  • ????????????for?(int?i?=?0;?i?<?segments.length;?++i)?{??
  • ????????????????int?c?=?segments[i].count;??
  • ????????????????if?(mc[i]?!=?segments[i].modCount)?{??
  • ????????????????????cleanSweep?=?false;??
  • ????????????????????break;??
  • ????????????????}??
  • ????????????}??
  • ????????}??
  • ????????if?(cleanSweep)??
  • ????????????return?false;??
  • ????}??
  • ????//?Resort?to?locking?all?segments??
  • ????for?(int?i?=?0;?i?<?segments.length;?++i)??
  • ????????segments[i].lock();??
  • ????boolean?found?=?false;??
  • ????try?{??
  • ????????for?(int?i?=?0;?i?<?segments.length;?++i)?{??
  • ????????????if?(segments[i].containsValue(value))?{??
  • ????????????????found?=?true;??
  • ????????????????break;??
  • ????????????}??
  • ????????}??
  • ????}?finally?{??
  • ????????for?(int?i?=?0;?i?<?segments.length;?++i)??
  • ????????????segments[i].unlock();??
  • ????}??
  • ????return?found;??
  • }??
  • ?

    同樣注意內(nèi)層的第一個for循環(huán),里面有語句int c = segments[i].count; 但是c卻從來沒有被使用過,即使如此,編譯器也不能做優(yōu)化將這條語句去掉,因為存在對volatile變量count的讀取,這條語句存在的唯一目的就是保證segments[i].modCount讀取到幾乎最新的值。關(guān)于containsValue方法的其它部分就不分析了,它和size方法差不多。

    ?

    ?

    跨段方法中還有一個isEmpty()方法,其實現(xiàn)比size()方法還要簡單,也不介紹了。最后簡單地介紹下迭代方法,如keySet(), values(), entrySet()方法,這些方法都返回相應(yīng)的迭代器,所有迭代器都繼承于Hash_Iterator類(提交時居然提醒我不能包含sh It,只得加了下劃線),里實現(xiàn)了主要的方法。其結(jié)構(gòu)是:

    ?

    Java代碼??
  • abstract?class?Hash_Iterator{??
  • ????int?nextSegmentIndex;??
  • ????int?nextTableIndex;??
  • ????HashEntry<K,V>[]?currentTable;??
  • ????HashEntry<K,?V>?nextEntry;??
  • ????HashEntry<K,?V>?lastReturned;??
  • }??
  • ?

    ?nextSegmentIndex是段的索引,nextTableIndex是nextSegmentIndex對應(yīng)段中中hash鏈的索引,currentTable是nextSegmentIndex對應(yīng)段的table。調(diào)用next方法時主要是調(diào)用了advance方法:

    ?

    ?

    Java代碼??
  • final?void?advance()?{??
  • ????if?(nextEntry?!=?null?&&?(nextEntry?=?nextEntry.next)?!=?null)??
  • ????????return;??
  • ??
  • ????while?(nextTableIndex?>=?0)?{??
  • ????????if?(?(nextEntry?=?currentTable[nextTableIndex--])?!=?null)??
  • ????????????return;??
  • ????}??
  • ??
  • ????while?(nextSegmentIndex?>=?0)?{??
  • ????????Segment<K,V>?seg?=?segments[nextSegmentIndex--];??
  • ????????if?(seg.count?!=?0)?{??
  • ????????????currentTable?=?seg.table;??
  • ????????????for?(int?j?=?currentTable.length?-?1;?j?>=?0;?--j)?{??
  • ????????????????if?(?(nextEntry?=?currentTable[j])?!=?null)?{??
  • ????????????????????nextTableIndex?=?j?-?1;??
  • ????????????????????return;??
  • ????????????????}??
  • ????????????}??
  • ????????}??
  • ????}??
  • }??
  • 不想再多介紹了,唯一需要注意的是跳到下一個段時,一定要先讀取下一個段的count變量。?

    ?

    這種迭代方式的主要效果是不會拋出ConcurrentModificationException。一旦獲取到下一個段的table,也就意味著這個段的頭結(jié)點在迭代過程中就確定了,在迭代過程中就不能反映對這個段節(jié)點并發(fā)的刪除和添加,對于節(jié)點的更新是能夠反映的,因為節(jié)點的值是一個volatile變量。

    ?


    結(jié)束語

    ?

    ConcurrentHashMap是一個支持高并發(fā)的高性能的HashMap實現(xiàn),它支持完全并發(fā)的讀以及一定程度并發(fā)的寫。ConcurrentHashMap的實現(xiàn)也是很精巧,充分利用了最新的JMM規(guī)范,值得學(xué)習(xí),卻不值得模仿。最后由于本人水平有限,對大師的作品難免有誤解,如果存在,還望大牛們不吝指出。

    ?

    ?

    ?

    ?

    參考文章:

    http://www.ibm.com/developerworks/java/library/j-jtp08223/,這個是討論的是Doug Lea's util.concurrent包中的ConcurrentHashMap的實現(xiàn),不過大致思想是一致的。

    ?

    http://floatingpoint.tinou.com/2008/09/performance-optimization-in-concurrenthashmap.html

    總結(jié)

    以上是生活随笔為你收集整理的ConcurrentHashMap之实现细节(转)的全部內(nèi)容,希望文章能夠幫你解決所遇到的問題。

    如果覺得生活随笔網(wǎng)站內(nèi)容還不錯,歡迎將生活随笔推薦給好友。

    主站蜘蛛池模板: 五月天激情影院 | 久久久久久久久久久久久久久久久久久 | 成人在线观看一区二区 | 四虎网址在线观看 | 色爽爽爽爽爽爽爽爽 | 免费天堂av | 国产最新视频 | 午夜视频福利网站 | 爱逼综合| 黄色网址在线免费观看 | 国产伦精品一区二区. | 国产天堂在线观看 | 天天做天天爱天天做 | 丝袜黄色片 | 97九色| 精品日日夜夜 | 国产中文一区 | 欧美a v在线 | 2020av视频 | 国产精品手机在线 | 粉嫩av一区二区三区四区五区 | 精品爱爱| 麻豆av一区二区三区 | 成人在线综合 | 日本少妇久久 | 中文字幕不卡在线观看 | 性久久久 | 人与拘一级a毛片 | 日本少妇一区二区 | 久草视频在线播放 | 黄瓜视频在线观看 | 影音先锋天堂网 | 国产精品自拍av | 69视频国产| 在线看国产视频 | 色窝窝无码一区二区三区 | 波多野结衣中文字幕久久 | 精品福利视频一区二区 | 蜜桃麻豆视频 | 在线看国产 | 日韩福利一区二区 | 欧美日韩一区二区三区四区五区 | 九九九久久久精品 | 欧美婷婷 | 国产美女无遮挡免费 | 国产日韩在线观看一区 | 在线免费观看亚洲 | 亚洲色图吧 | 日韩大尺度视频 | 成人a毛片久久免费播放 | 91精品在线免费观看 | 色网导航站 | 97久久免费视频 | 天天干天天操天天干 | 国产人妻人伦精品1国产盗摄 | 国产精欧美一区二区三区白种人 | 日本乱码一区二区 | 少妇久久久久 | 色猫咪av| 爱插视频 | 亚洲性生活 | 中文字幕在线观看高清 | 四虎新网址| 日韩不卡av | 国产精品伦一区二区三区免费看 | 亚洲成人精品av | 色桃av| 91精品久久久久久综合五月天 | 手机在线看永久av片免费 | 国产无遮无挡120秒 欧美综合图片 | 福利所导航 | 一区二区免费在线视频 | www射| 欧美视频免费在线观看 | 青青草伊人网 | 日韩av在线不卡 | 日本一级片在线观看 | 欧美成人毛片 | 亚洲一区二区免费电影 | 久久久精品一区二区三区 | 欧美片在线观看 | av第一福利| 五月天激情开心网 | v片在线免费观看 | 久久888 | 日韩电影在线一区 | 久久久久成人精品无码中文字幕 | 日韩色婷婷 | 国产欧美日韩精品在线观看 | 蜜色视频 | 蜜臀av88| 一区免费在线 | 免费av手机在线观看 | 我爱我色成人网 | 亚洲伊人久久综合 | 亚洲性网 | 朋友人妻少妇精品系列 | 国产高中女学生第一次 | 欧美精品性生活 |