<em>Mac</em>Book项目 2009年学校开始实施<em>Mac</em>Book项目,所有师生配备一本<em>Mac</em>Book,并同步更新了校园无线网络。学校每周进行电脑技术更新,每月发送技术支持资料,极大改变了教学及学习方式。因此2011
2021-06-01 09:32:01
為了方便編寫出執行緒安全的程式,Java裡面提供了一些執行緒安全類和並行工具,比如:同步容器、並行容器、阻塞佇列等。
最常見的同步容器就是Vector和Hashtable了,那麼,同步容器的所有操作都是執行緒安全的嗎?下面我們來一一分析這個問題。
在Java中,同步容器主要包括2類:
我們以相對簡單的Vecotr來舉例,我們先來看下Vector中幾個重要方法的原始碼:
public synchronized boolean add(E e) { modCount++; ensureCapacityHelper(elementCount + 1); elementData[elementCount++] = e; return true; } public synchronized E remove(int index) { modCount++; if (index >= elementCount) throw new ArrayIndexOutOfBoundsException(index); E oldValue = elementData(index); int numMoved = elementCount - index - 1; if (numMoved > 0) System.arraycopy(elementData, index+1, elementData, index, numMoved); elementData[--elementCount] = null; // Let gc do its work return oldValue; } public synchronized E get(int index) { if (index >= elementCount) throw new ArrayIndexOutOfBoundsException(index); return elementData(index); }
可以看到,Vector這樣的同步容器的所有公有方法全都是synchronized的,也就是說,我們可以在多執行緒場景中放心的使用單獨這些方法,因為這些方法本身的確是執行緒安全的。
但是,請注意上面這句話中,有一個比較關鍵的詞:單獨
因為,雖然同步容器的所有方法都加了鎖,但是對這些容器的複合操作無法保證其執行緒安全性。需要使用者端通過主動加鎖來保證。
簡單舉一個例子,我們定義如下刪除Vector中最後一個元素方法:
public Object deleteLast(Vector v){ int lastIndex = v.size()-1; v.remove(lastIndex); }
上面這個方法是一個複合方法,包括 size()和 remove(),看上去好像並沒有什麼問題,無論是size()方法還是remove()方法都是執行緒安全的,那麼整個deleteLast方法應該也是執行緒安全的。
但是,如果多執行緒呼叫該方法的過程中,remove方法有可能丟擲ArrayIndexOutOfBoundsException:
Exception in thread "Thread-1" java.lang.ArrayIndexOutOfBoundsException: Array index out of range: 879 at java.util.Vector.remove(Vector.java:834) at com.hollis.Test.deleteLast(EncodeTest.java:40) at com.hollis.Test$2.run(EncodeTest.java:28) at java.lang.Thread.run(Thread.java:748)
我們上面貼了remove的原始碼,我們可以分析得出:當index >= elementCount時,會丟擲ArrayIndexOutOfBoundsException ,也就是說,噹噹前索引值不再有效的時候,將會丟擲這個異常。
因為removeLast方法,有可能被多個執行緒同時執行,當執行緒2通過index()獲得索引值為10,在嘗試通過remove()刪除該索引位置的元素之前,執行緒1把該索引位置的值刪除掉了,這時執行緒一在執行時便會丟擲異常。
為了避免出現類似問題,可以嘗試加鎖:
public void deleteLast() { synchronized (v) { int index = v.size() - 1; v.remove(index); } }
如上,我們在deleteLast中,對v進行加鎖,即可保證同一時刻,不會有其他執行緒刪除掉v中的元素。
另外,如果以下程式碼會被多執行緒執行時,也要特別注意:
for (int i = 0; i < v.size(); i++) { v.remove(i); }
由於,不同執行緒在同一時間操作同一個Vector,其中包括刪除操作,那麼就同樣有可能發生執行緒安全問題。所以,在使用同步容器的時候,如果涉及到多個執行緒同時執行刪除操作,就要考慮下是否需要加鎖。
前面說過了,同步容器直接保證單個操作的執行緒安全性,但是無法保證複合操作的執行緒安全,遇到這種情況時,必須要通過主動加鎖的方式來實現。
而且,除此之外,由於所有方法都加了鎖,這就導致多個執行緒存取同一個容器的時候,只能進行順序存取,即使是不同的操作,也要排隊,如get和add要排隊執行。這就大大的降低了容器的並行能力。
針對前文提到的同步容器存在的並行度低問題,從Java5開始,java.util.concurent包下,提供了大量支援高效並行的存取的集合類,我們稱之為並行容器。
針對前面提到的同步容器的複合操作的問題,一般在 Map 中發生的比較多,所以在ConcurrentHashMap中增加了對常用複合操作的支援,比如putIfAbsent()、replace(),這2個操作都是原子操作,可以保證執行緒安全。
另外,並行包中的CopyOnWriteArrayList和CopyOnWriteArraySet是Copy-On-Write的兩種實現。
Copy-On-Write容器即寫時複製的容器。通俗的理解是當我們往一個容器新增元素的時候,不直接往當前容器新增,而是先將當前容器進行Copy,複製出一個新的容器,然後新的容器裡新增元素,新增完元素之後,再將原容器的參照指向新的容器。
CopyOnWriteArrayList中add/remove等寫方法是需要加鎖的,而讀方法是沒有加鎖的。
這樣做的好處是我們可以對CopyOnWrite容器進行並行的讀,當然,這裡讀到的資料可能不是最新的。因為寫時複製的思想是通過延時更新的策略來實現資料的最終一致性的,並非強一致性。
但是,作為代替Vector的CopyOnWriteArrayList並沒有解決同步容器的複合操作的執行緒安全性問題。
以上就是本文的全部內容,希望對大家的學習有所幫助,也希望大家多多支援it145.com。
相關文章
<em>Mac</em>Book项目 2009年学校开始实施<em>Mac</em>Book项目,所有师生配备一本<em>Mac</em>Book,并同步更新了校园无线网络。学校每周进行电脑技术更新,每月发送技术支持资料,极大改变了教学及学习方式。因此2011
2021-06-01 09:32:01
综合看Anker超能充系列的性价比很高,并且与不仅和iPhone12/苹果<em>Mac</em>Book很配,而且适合多设备充电需求的日常使用或差旅场景,不管是安卓还是Switch同样也能用得上它,希望这次分享能给准备购入充电器的小伙伴们有所
2021-06-01 09:31:42
除了L4WUDU与吴亦凡已经多次共事,成为了明面上的厂牌成员,吴亦凡还曾带领20XXCLUB全队参加2020年的一场音乐节,这也是20XXCLUB首次全员合照,王嗣尧Turbo、陈彦希Regi、<em>Mac</em> Ova Seas、林渝植等人全部出场。然而让
2021-06-01 09:31:34
目前应用IPFS的机构:1 谷歌<em>浏览器</em>支持IPFS分布式协议 2 万维网 (历史档案博物馆)数据库 3 火狐<em>浏览器</em>支持 IPFS分布式协议 4 EOS 等数字货币数据存储 5 美国国会图书馆,历史资料永久保存在 IPFS 6 加
2021-06-01 09:31:24
开拓者的车机是兼容苹果和<em>安卓</em>,虽然我不怎么用,但确实兼顾了我家人的很多需求:副驾的门板还配有解锁开关,有的时候老婆开车,下车的时候偶尔会忘记解锁,我在副驾驶可以自己开门:第二排设计很好,不仅配置了一个很大的
2021-06-01 09:30:48
不仅是<em>安卓</em>手机,苹果手机的降价力度也是前所未有了,iPhone12也“跳水价”了,发布价是6799元,如今已经跌至5308元,降价幅度超过1400元,最新定价确认了。iPhone12是苹果首款5G手机,同时也是全球首款5nm芯片的智能机,它
2021-06-01 09:30:45