亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb

首頁 > 學院 > 開發設計 > 正文

集合是編程中最常用的數據結構

2019-11-06 09:23:18
字體:
來源:轉載
供稿:網友

集合是編程中最常用的數據結構。而談到并發,幾乎總是離不開集合這類高級數據結構的支持。比如兩個線程需要同時訪問一個中間臨界區(Queue),比如常會用緩存作為外部文件的副本(HashMap)。這篇文章主要分析jdk1.5的3種并發集合類型(concurrent,copyonright,queue)中的ConcurrentHashMap,讓我們從原理上細致的了解它們,能夠讓我們在深度項目開發中獲益非淺。

    通過分析Hashtable就知道,synchronized是針對整張Hash表的,即每次鎖住整張表讓線程獨占,ConcurrentHashMap允許多個修改操作并發進行,其關鍵在于使用了鎖分離技術。它使用了多個鎖來控制對hash表的不同部分進行的修改。ConcurrentHashMap內部使用段(Segment)來表示這些不同的部分,每個段其實就是一個小的hash table,它們有自己的鎖。只要多個修改操作發生在不同的段上,它們就可以并發進行。有些方法需要跨段,比如size()和containsValue(),它們可能需要鎖定整個表而而不僅僅是某個段,這需要按順序鎖定所有段,操作完畢后,又按順序釋放所有段的鎖。這里“按順序”是很重要的,否則極有可能出現死鎖,在ConcurrentHashMap內部,段數組是final的,并且其成員變量實際上也是final的,但是,僅僅是將數組聲明為final的并不保證數組成員也是final的,這需要實現上的保證。這可以確保不會出現死鎖,因為獲得鎖的順序是固定的。

 一、結構解析

   ConcurrentHashMap和Hashtable主要區別就是圍繞著鎖的粒度以及如何鎖,可以簡單理解成把一個大的HashTable分解成多個,形成了鎖分離。如圖:

而Hashtable的實現方式是---鎖整個hash表

二、應用場景

當有一個大數組時需要在多個線程共享時就可以考慮是否把它給分層多個節點了,避免大鎖。并可以考慮通過hash算法進行一些模塊定位。

其實不止用于線程,當設計數據表的事務時(事務某種意義上也是同步機制的體現),可以把一個表看成一個需要同步的數組,如果操作的表數據太多時就可以考慮事務分離了(這也是為什么要避免大表的出現),比如把數據進行字段拆分,水平分表等.

三、源碼解讀

 ConcurrentHashMap中主要實體類就是三個:ConcurrentHashMap(整個Hash表),Segment(桶),HashEntry(節點),對應上面的圖可以看出之間的關系

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

不變(Immutable)和易變(Volatile)ConcurrentHashMap完全允許多個讀操作并發進行,讀操作并不需要加鎖。如果使用傳統的技術,如HashMap中的實現,如果允許可以在hash鏈的中間添加或刪除元素,讀操作不加鎖將得到不一致的數據。ConcurrentHashMap實現技術是保證HashEntry幾乎是不可變的。HashEntry代表每個hash鏈中的一個節點,其結構如下所示:

 

1. static final class HashEntry<K,V> {  2.     final K key;  3.     final int hash;  4.     volatile V value;  5.     final HashEntry<K,V> next;  6. }  

 

可以看到除了value不是final的,其它值都是final的,這意味著不能從hash鏈的中間或尾部添加或刪除節點,因為這需要修改next 引用值,所有的節點的修改只能從頭部開始。對于put操作,可以一律添加到Hash鏈的頭部。但是對于remove操作,可能需要從中間刪除一個節點,這就需要將要刪除節點的前面所有節點整個復制一遍,最后一個節點指向要刪除結點的下一個結點。這在講解刪除操作時還會詳述。為了確保讀操作能夠看到最新的值,將value設置成volatile,這避免了加鎖。其它為了加快定位段以及段中hash槽的速度,每個段hash槽的的個數都是2^n,這使得通過位運算就可以定位段和段中hash槽的位置。當并發級別為默認值16時,也就是段的個數,hash值的高4位決定分配在哪個段中。但是我們也不要忘記《算法導論》給我們的教訓:hash槽的的個數不應該是 2^n,這可能導致hash槽分配不均,這需要對hash值重新再hash一次。(這段似乎有點多余了 )

 

這是定位段的方法:

1. final Segment<K,V> segmentFor(int hash) {  2.     return segments[(hash >>> segmentShift) & segmentMask];  3. } 

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

 

復制代碼復制代碼
1. public class ConcurrentHashMap<K, V> extends AbstractMap<K, V>  2.         implements ConcurrentMap<K, V>, Serializable {  3.     /** 4.      * Mask value for indexing into segments. The upper bits of a 5.      * key's hash code are used to choose the segment. 6.      */  7.     final int segmentMask;  8.   9.     /** 10.      * Shift value for indexing within segments. 11.      */  12.     final int segmentShift;  13.   14.     /** 15.      * The segments, each of which is a specialized hash table 16.      */  17.     final Segment<K,V>[] segments;  18. } 復制代碼復制代碼

 

所有的成員都是final的,其中segmentMask和segmentShift主要是為了定位段,參見上面的segmentFor方法。每個Segment相當于一個子Hash表,它的數據成員如下:

 

復制代碼復制代碼
1.     static final class Segment<K,V> extends ReentrantLock implements Serializable {  2. PRivate static final long serialVersionUID = 2249069246763182397L;  3.         /** 4.          * The number of elements in this segment's region. 5.          */  6.         transient volatile int count;  7.   8.         /** 9.          * Number of updates that alter the size of the table. This is 10.          * used during bulk-read methods to make sure they see a 11.          * consistent snapshot: If modCounts change during a traversal 12.          * of segments computing size or checking containsValue, then 13.          * we might have an inconsistent view of state so (usually) 14.          * must retry. 15.          */  16.         transient int modCount;  17.   18.         /** 19.          * The table is rehashed when its size exceeds this threshold. 20.          * (The value of this field is always <tt>(int)(capacity * 21.          * loadFactor)</tt>.) 22.          */  23.         transient int threshold;  24.   25.         /** 26.          * The per-segment table. 27.          */  28.         transient volatile HashEntry<K,V>[] table;  29.   
http://www.kuaimin.cn/zonghe/20170227/293598.html
http://www.kuaimin.cn/zonghe/20170227/293597.html
http://www.kuaimin.cn/zonghe/20170227/293590.html
30. /** 31. * The load factor for the hash table. Even though this value 32. * is same for all segments, it is replicated to avoid needing 33. * links to outer object. 34. * @serial 35. */ 36. final float loadFactor; 37. } 復制代碼復制代碼

 

count用來統計該段數據的個數,它是volatile(volatile 變量使用指南),它用來協調修改和讀取操作,以保證讀取操作能夠讀取到幾乎最新的修改。協調方式是這樣的,每次修改操作做了結構上的改變,如增加/刪除節點(修改節點的值不算結構上的改變),都要寫count值,每次讀取操作開始都要讀取count的值。這利用了 java 5中對volatile語義的增強,對同一個volatile變量的寫和讀存在happens-before關系。modCount統計段結構改變的次數,主要是為了檢測對多個段進行遍歷過程中某個段是否發生改變,在講述跨段操作時會還會詳述。threashold用來表示需要進行rehash的界限值。table數組存儲段中節點,每個數組元素是個hash鏈,用HashEntry表示。table也是volatile,這使得能夠讀取到最新的 table值而不需要同步。loadFactor表示負載因子。


發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb
一本大道久久加勒比香蕉| 成人精品一区二区三区电影免费| 国产精品va在线播放| 日日骚久久av| 国产精品偷伦免费视频观看的| 欧美精品18videos性欧| 在线亚洲男人天堂| 中文字幕亚洲色图| 这里只有视频精品| 欧美成aaa人片免费看| 精品久久在线播放| 欧美情侣性视频| 国产成人精彩在线视频九色| 国产69精品久久久| 日韩av在线网页| 亚洲精品欧美日韩专区| 国产国产精品人在线视| 国产精品视频区| 中文字幕免费国产精品| 久久久久久久色| 亚洲欧洲午夜一线一品| 大量国产精品视频| 精品亚洲永久免费精品| 亚洲国产一区二区三区在线观看| 国产在线98福利播放视频| 两个人的视频www国产精品| 日韩最新中文字幕电影免费看| 8050国产精品久久久久久| 欧美国产日韩免费| 性欧美视频videos6一9| 亚洲性xxxx| 成人午夜在线观看| 成人免费在线视频网站| 亚洲国产精彩中文乱码av| 欧美高清电影在线看| 久久精品人人爽| 国内精品小视频在线观看| 亚洲精品国产综合区久久久久久久| 中文字幕在线成人| 国产69精品久久久久久| 精品国产一区二区三区久久| 国产精品91久久久久久| 成人黄色激情网| 欧美日韩激情网| 国产成人激情小视频| 国产日韩欧美在线视频观看| 狠狠综合久久av一区二区小说| 超薄丝袜一区二区| 夜夜嗨av色综合久久久综合网| 成人在线一区二区| 精品视频偷偷看在线观看| 性色av一区二区三区红粉影视| 国产在线视频欧美| 欧美性生交xxxxx久久久| 亚洲人成在线免费观看| 日本精品久久久| 欧美在线观看一区二区三区| 国产成人aa精品一区在线播放| www.久久久久久.com| 全色精品综合影院| 98精品国产自产在线观看| 国产福利精品在线| 久久久久国产精品www| 欧美专区在线观看| 国产精品69精品一区二区三区| 欧美激情国产日韩精品一区18| 久久精品电影网| 日韩欧美一区二区在线| 亚洲图片欧洲图片av| 色妞欧美日韩在线| 久久久亚洲国产天美传媒修理工| 亚洲高清免费观看高清完整版| 按摩亚洲人久久| 日本久久久久久久久久久| 国产成人精品久久| 欧美日韩视频免费播放| 成人av在线网址| 高跟丝袜欧美一区| 欧美黑人又粗大| 成人精品aaaa网站| 亚洲人成电影网站色…| 日韩风俗一区 二区| 欧美另类极品videosbestfree| 91伊人影院在线播放| 91精品国产91久久久久福利| www高清在线视频日韩欧美| 日本国产一区二区三区| 欧美日韩综合视频| 国产精品久久久久久久久久小说| 国产精品白丝av嫩草影院| 中文字幕一区二区三区电影| 亚洲自拍高清视频网站| 91福利视频网| 日韩欧美精品网址| 国产做受高潮69| 欧美日韩国产91| 日韩欧美高清视频| 久久成人亚洲精品| 欧美性在线观看| 2020久久国产精品| 欧美黄色片在线观看| 久久久久久欧美| 欧美与欧洲交xxxx免费观看| 日韩高清av一区二区三区| 国产999精品| 欧美日韩福利视频| 欧美日韩精品中文字幕| 亚洲欧美日本伦理| 亚洲自拍小视频| 久久激情视频免费观看| 亚洲www永久成人夜色| 国产福利精品av综合导导航| 国产精品日韩一区| 色偷偷888欧美精品久久久| 日韩美女主播视频| 91在线免费观看网站| 北条麻妃久久精品| 国产精品欧美日韩| 91欧美激情另类亚洲| 久久久www成人免费精品张筱雨| 久久久久亚洲精品成人网小说| 亚洲在线一区二区| 亚洲激情在线观看视频免费| www.日本久久久久com.| 欧美日韩福利在线观看| 久久精品美女视频网站| 欧美另类第一页| 欧美性受xxxx黑人猛交| 亚洲男人天堂九九视频| 91精品久久久久久久久| 欧美日韩一二三四五区| 精品偷拍一区二区三区在线看| 欧美性高潮在线| 国产精品444| 国产精品激情av在线播放| 国产精品福利在线观看网址| 亚洲欧美日韩一区在线| 精品视频久久久| 国产一区二区三区欧美| 久久久久久亚洲| 国产精品成人观看视频国产奇米| 一本久久综合亚洲鲁鲁| 亚洲欧美日韩中文视频| 性欧美xxxx视频在线观看| 中文字幕亚洲欧美| 国产精品视频一区国模私拍| 中文字幕亚洲无线码在线一区| 亚洲欧美国产另类| 美女扒开尿口让男人操亚洲视频网站| 久热精品视频在线观看一区| 亚洲黄在线观看| 欧美视频一二三| 色yeye香蕉凹凸一区二区av| 亚洲国产精品va在线看黑人| 日韩a**站在线观看| 91国偷自产一区二区三区的观看方式| 亚洲最大福利视频网| 欧美激情亚洲国产| 久久精品中文字幕免费mv| 欧美激情亚洲一区| 国产精品久久久久久av福利| 欧美激情第6页| 亚洲精品久久久久| 性色av香蕉一区二区|