往map里的vector添加_面试官问我同步容器(如Vector)的所有操作一定是线程安全的吗?我懵了!...
生活随笔
收集整理的這篇文章主要介紹了
往map里的vector添加_面试官问我同步容器(如Vector)的所有操作一定是线程安全的吗?我懵了!...
小編覺得挺不錯的,現在分享給大家,幫大家做個參考.
為了方便編寫出線程安全的程序,Java里面提供了一些線程安全類和并發工具,比如:同步容器、并發容器、阻塞隊列等。
最常見的同步容器就是Vector和Hashtable了,那么,同步容器的所有操作都是線程安全的嗎?
這個問題不知道你有沒有想過,本文就來深入分析一下這個問題,一個很容易被忽略的問題。
1
同步容器
在Java中,同步容器主要包括2類:
- 1、Vector、Stack、HashTable
- 2、Collections類中提供的靜態工廠方法創建的類
2
同步容器的問題
前面說過了,同步容器直接保證單個操作的線程安全性,但是無法保證復合操作的線程安全,遇到這種情況時,必須要通過主動加鎖的方式來實現。而且,除此之外,同步容易由于對其所有方法都加了鎖,這就導致多個線程訪問同一個容器的時候,只能進行順序訪問,即使是不同的操作,也要排隊,如get和add要排隊執行。這就大大的降低了容器的并發能力。3
并發容器
針對前文提到的同步容器存在的并發度低問題,從Java5開始,java.util.concurent包下,提供了大量支持高效并發的訪問的集合類,我們稱之為并發容器。針對前文提到的同步容器的復合操作的問題,一般在Map中發生的比較多,所以在ConcurrentHashMap中增加了對常用復合操作的支持,比如putIfAbsent()、replace(),這2個操作都是原子操作,可以保證線程安全。另外,并發包中的CopyOnWriteArrayList和CopyOnWriteArraySet是Copy-On-Write的兩種實現。Copy-On-Write容器即寫時復制的容器。通俗的理解是當我們往一個容器添加元素的時候,不直接往當前容器添加,而是先將當前容器進行Copy,復制出一個新的容器,然后新的容器里添加元素,添加完元素之后,再將原容器的引用指向新的容器。CopyOnWriteArrayList中add/remove等寫方法是需要加鎖的,而讀方法是沒有加鎖的。這樣做的好處是我們可以對CopyOnWrite容器進行并發的讀,當然,這里讀到的數據可能不是最新的。因為寫時復制的思想是通過延時更新的策略來實現數據的最終一致性的,并非強一致性。但是,作為代替Vector的CopyOnWriteArrayList并沒有解決同步容器的復合操作的線程安全性問題。4
總結
本文介紹了同步容器和并發容器。同步容器是通過加鎖實現線程安全的,并且只能保證單獨的操作是線程安全的,無法保證復合操作的線程安全性。并且同步容器的讀和寫操作之間會互相阻塞。并發容器是Java 5中提供的,主要用來代替同步容器。有更好的并發能力。而且其中的ConcurrentHashMap定義了線程安全的復合操作。在多線程場景中,如果使用并發容器,一定要注意復合操作的線程安全問題。必要時候要主動加鎖。在并發場景中,建議直接使用java.util.concurent包中提供的容器類,如果需要復合操作時,建議使用有些容器自身提供的復合方法。有道無術,術可成;有術無道,止于術
歡迎大家關注Java之道公眾號
好文章,我在看??
總結
以上是生活随笔為你收集整理的往map里的vector添加_面试官问我同步容器(如Vector)的所有操作一定是线程安全的吗?我懵了!...的全部內容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: 互联网技术概念入门
- 下一篇: 如何设置MySQL的环境变量