在Java併發編程中,常常聽到Java集合類,同步容器、併發容器,那麼他們有哪些具體分類,以及各自之間的區別和優劣呢?java
只有把這些梳理清楚了,你才能真正掌握在高併發的環境下,正確使用好併發容器,咱們先從Java集合類,同步容器談起。程序員
Java的集合容器框架中,主要有四大類別:List、Set、Queue、Map,你們熟知的這些集合類ArrayList、LinkedList、HashMap這些容器都是非線程安全的。算法
若是有多個線程併發地訪問這些容器時,就會出現問題。所以,在編寫程序時,在多線程環境下必需要求程序員手動地在任何訪問到這些容器的地方進行同步處理,這樣致使在使用這些容器的時候很是地不方便。編程
因此,Java先提供了同步容器供用戶使用。segmentfault
同步容器能夠簡單地理解爲經過synchronized來實現同步的容器,好比Vector、Hashtable以及SynchronizedList等容器。數組
同步容器面臨的問題安全
能夠經過查看Vector,Hashtable等這些同步容器的實現代碼,能夠看到這些容器實現線程安全的方式就是將它們的狀態封裝起來,並在須要同步的方法上加上關鍵字synchronized。數據結構
這樣作的代價是削弱了併發性,當多個線程共同競爭容器級的鎖時,吞吐量就會下降。多線程
例如: HashTable只要有一條線程獲取了容器的鎖以後,其餘全部的線程訪問同步函數都會被阻塞,所以同一時刻只能有一條線程訪問同步函數。併發
所以爲了解決同步容器的性能問題,因此纔有了併發容器。
java.util.concurrent包中提供了多種併發類容器。
併發類容器是專門針對多線程併發設計的,使用了鎖分段技術,只對操做的位置進行同步操做,可是其餘沒有操做的位置其餘線程仍然能夠訪問,提升了程序的吞吐量。
採用了CAS算法和部分代碼使用synchronized鎖保證線程安全。
1.ConcurrentHashMap
對應的非併發容器:HashMap
目標:代替Hashtable、synchronizedMap,支持複合操做
原理:JDK6中採用一種更加細粒度的加鎖機制Segment「分段鎖」,JDK8中採用CAS無鎖算法。
2.CopyOnWriteArrayList
對應的非併發容器:ArrayList
目標:代替Vector、synchronizedList
原理:利用高併發每每是讀多寫少的特性,對讀操做不加鎖,對寫操做,先複製一份新的集合,在新的集合上面修改,而後將新集合賦值給舊的引用,並經過volatile 保證其可見性,固然寫操做的鎖是必不可少的了。
3.CopyOnWriteArraySet
對應的非併發容器:HashSet
目標:代替synchronizedSet
原理:基於CopyOnWriteArrayList實現,其惟一的不一樣是在add時調用的是CopyOnWriteArrayList的addIfAbsent方法,其遍歷當前Object數組,如Object數組中已有了當前元素,則直接返回,若是沒有則放入Object數組的尾部,並返回。
4.ConcurrentSkipListMap
對應的非併發容器:TreeMap
目標:代替synchronizedSortedMap(TreeMap)
原理:Skip list(跳錶)是一種能夠代替平衡樹的數據結構,默認是按照Key值升序的。
5.ConcurrentSkipListSet
對應的非併發容器:TreeSet
目標:代替synchronizedSortedSet
原理:內部基於ConcurrentSkipListMap實現
6.ConcurrentLinkedQueue
不會阻塞的隊列
對應的非併發容器:Queue
原理:基於鏈表實現的FIFO隊列(LinkedList的併發版本)
7.LinkedBlockingQueue、ArrayBlockingQueue、PriorityBlockingQueue
對應的非併發容器:BlockingQueue
特色:拓展了Queue,增長了可阻塞的插入和獲取等操做
原理:經過ReentrantLock實現線程安全,經過Condition實現阻塞和喚醒
實現類:
HashMap,Hashtable與ConcurrentHashMap都是實現的哈希表數據結構,在隨機讀取的時候效率很高。
Hashtable實現同步是利用synchronized關鍵字進行鎖定的,其是針對整張哈希表進行鎖定的,即每次鎖住整張表讓線程獨佔,在線程安全的背後是巨大的浪費。
ConcurrentHashMap和Hashtable主要區別就是圍繞着鎖的粒度進行區別以及如何區鎖定。
上圖中,左邊是Hashtable的實現方式,能夠看到鎖住整個哈希表;而右邊則是ConcurrentHashMap的實現方式,單獨鎖住每個桶(segment).ConcurrentHashMap將哈希表分爲16個桶(默認值),諸如get(),put(),remove()等經常使用操做只鎖當前須要用到的桶,而size()才鎖定整張表。
原來只能一個線程進入,如今卻能同時接受16個寫線程併發進入(寫線程須要鎖定,而讀線程幾乎不受限制)。
因此,纔有了併發性的極大提高。
高併發編程,除了併發容器,還會涉及到併發工具類:CountDownLatch等,後續將詳細的介紹併發工具類,以及ConcurrentHashMap的底層實現細節,不只要知其然,還要知其因此然,這樣才能更好的掌握好高併發編程。