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

歡迎訪問 生活随笔!

生活随笔

當前位置: 首頁 > 编程资源 > 编程问答 >内容正文

编程问答

原创 | 面试官问我同步容器(如Vector)的所有操作一定是线程安全的吗?我懵了!...

發布時間:2025/3/16 编程问答 18 豆豆
生活随笔 收集整理的這篇文章主要介紹了 原创 | 面试官问我同步容器(如Vector)的所有操作一定是线程安全的吗?我懵了!... 小編覺得挺不錯的,現在分享給大家,幫大家做個參考.

△Hollis, 一個對Coding有著獨特追求的人△

這是Hollis的第?226篇原創分享

作者 l Hollis

來源 l Hollis(ID:hollischuang)

為了方便編寫出線程安全的程序,Java里面提供了一些線程安全類和并發工具,比如:同步容器、并發容器、阻塞隊列等。

最常見的同步容器就是Vector和Hashtable了,那么,同步容器的所有操作都是線程安全的嗎?

這個問題不知道你有沒有想過,本文就來深入分析一下這個問題,一個很容易被忽略的問題。

1

同步容器

在Java中,同步容器主要包括2類:

  • 1、Vector、Stack、HashTable

  • 2、Collections類中提供的靜態工廠方法創建的類

本文拿相對簡單的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,其中包括刪除操作,那么就同樣有可能發生線程安全問題。所以,在使用同步容器的時候,如果涉及到多個線程同時執行刪除操作,就要考慮下是否需要加鎖。

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包中提供的容器類,如果需要復合操作時,建議使用有些容器自身提供的復合方法。

快薅,當當的羊毛,晚了就沒了!

快薅,當當的羊毛,晚了就沒了!

快薅,當當的羊毛,晚了就沒了!



- MORE | 更多精彩文章 -

  • 華為發布會: ?牛逼鴻蒙,吹水的大會

  • 我被失聯2年后,終于從東南亞的技術“魔窟”逃出來了...

  • 如何參與一個頂級開源項目

  • 原創 | 既生synchronized,何生volatile?!


如果你喜歡本文,

請長按二維碼,關注?Hollis.

轉發至朋友圈,是對我最大的支持。

好文章,我在看??

總結

以上是生活随笔為你收集整理的原创 | 面试官问我同步容器(如Vector)的所有操作一定是线程安全的吗?我懵了!...的全部內容,希望文章能夠幫你解決所遇到的問題。

如果覺得生活随笔網站內容還不錯,歡迎將生活随笔推薦給好友。

主站蜘蛛池模板: 欧美aa视频 | 国内外成人免费视频 | 国产色av| 久在线| 黄色一极毛片 | 亚洲精品国产精品乱码视色 | 午夜嘿嘿 | 日韩av在线观看免费 | 国产精品扒开腿做爽爽 | 久久最新网址 | 国产福利91精品一区二区三区 | 久色视频在线播放 | 99久久久国产精品无码免费 | 美女一区二区三区四区 | 亚洲区一区二 | 欧洲亚洲女同hd | 精品日韩视频 | 污免费视频 | 天天射天天操天天干 | 久久综合亚洲 | caopor超碰| 国产免费av片在线 | 女王脚交玉足榨精调教 | 成人欧美一区二区三区黑人一 | 日韩在线视频免费看 | 五月婷婷丁香六月 | 禁漫天堂在线 | 视频福利在线观看 | 国产高清久久久 | 深夜福利免费视频 | 龚玥菲三级露全乳视频 | 国产成人无码性教育视频 | 大陆农村乡下av | 欧美乱妇15p | 国内av在线播放 | 免费看黄色片的网站 | 国产成人在线观看免费网站 | 就是色| 日韩av高清无码 | 精品无码一区二区三区在线 | 神马午夜场 | 日韩精品区 | 亚洲一线在线观看 | 久久综合久久鬼色 | 污视频免费看 | 日韩av中文字幕在线 | 男女啪啪免费网站 | 亚洲熟妇无码爱v在线观看 九色福利 | 91av视频在线免费观看 | 国产精品一二区在线观看 | 欧美福利视频在线观看 | 天堂8中文在线 | 黑人操日本女人 | 亚洲福利 | 国产精品久久中文字幕 | 一级片aaaaa 国产又黄又粗又猛又爽 | 凹凸视频一区二区 | 歪歪视频在线观看 | 久久夜夜夜 | www.三级.com | 男人猛进女人爽的大叫 | 黄页嫩草 | 亚洲社区一区二区 | 婷婷国产一区 | 日本老肥婆bbbwbbbwzr | 内地级a艳片高清免费播放 91在线精品一区二区 | 草草影院ccyycom | av免费的 | 成人国产一区二区三区 | 尤物视频在线观看国产 | 中文字幕一区二区三区门四区五区 | 国产主播啪啪 | 欧美在线免费视频 | tube国产麻豆 | 激情福利网 | xxx性日本 | 色婷在线 | 色婷婷在线影院 | 国产影视一区二区三区 | 亚洲AV永久无码国产精品国产 | 中文字幕在线视频第一页 | 男人的天堂免费 | 欧美 日韩 国产 一区二区三区 | 亚洲成人播放 | 久久久久久久亚洲精品 | 捆绑中国女人hd视频 | 成人小视频在线免费观看 | 四虎黄色网| 欧美亚洲图片小说 | 波多野结衣福利 | 狠狠干夜夜干 | 女人下边被添全过视频 | 91网站大全| 久久久久亚洲色欲AV无码网站 | 日韩国产欧美一区二区三区 | 一区二区在线免费 | 天堂va蜜桃一区二区三区漫画版 | 亚洲天堂免费视频 | 香蕉视频在线免费看 |