色综合图-色综合图片-色综合图片二区150p-色综合图区-玖玖国产精品视频-玖玖香蕉视频

您的位置:首頁技術(shù)文章
文章詳情頁

java的各種集合為什么不安全(List、Set、Map)以及代替方案

瀏覽:3日期:2022-08-22 08:41:40

我們已經(jīng)知道多線程下會有各種不安全的問題,都知道并發(fā)的基本解決方案,這里對出現(xiàn)錯誤的情況進(jìn)行一個實(shí)際模擬,以此能夠聯(lián)想到具體的生產(chǎn)環(huán)境中。

一、List 的不安全

1.1 問題

看一段代碼:

public static void main(String[] args) { ArrayList<String> list = new ArrayList<>(); for (int i = 0; i < 3; i++){ new Thread(()->{ list.add(UUID.randomUUID().toString().substring(0,8)); System.out.println(list); },String.valueOf(i)).start(); }}

過程很簡單,只有 3 個線程而已,對同一個 list 進(jìn)行 add 的寫操作,并隨后進(jìn)行輸出的讀操作。

輸出結(jié)果,多執(zhí)行幾次,驚喜多多。

java的各種集合為什么不安全(List、Set、Map)以及代替方案

那么,情況不嚴(yán)重的時候,這里顯然還正常運(yùn)行結(jié)束了,只是導(dǎo)致了還沒來得及寫的時候,就已經(jīng)讀出了數(shù)據(jù)。

如果把線程數(shù)增加試試,可能還會看到這樣的奇觀:

java的各種集合為什么不安全(List、Set、Map)以及代替方案

報(bào)錯了:重點(diǎn)異常:java.util.ConcurrentModificationException,翻譯過來就是并發(fā)修改異常。

1.2 產(chǎn)生原因

普通的 ArrayList 集合里面沒有任何特殊處理,在多線程情況下,他們可以共同進(jìn)行訪問。

那么在多線程同時操作的時候,按照操作的情況就有這幾種:

各個線程都讀。不影響,前提是只有讀;

各個線程都寫。會出現(xiàn)問題,這里的點(diǎn)有兩種情況:

值覆蓋問題,因?yàn)?ArrayList 的底層數(shù)組,寫入值的時候要先計(jì)算到一個下標(biāo)位置,然后給對應(yīng)的位置去賦值,多線程就會出現(xiàn)值覆蓋的問題; 空指針異常,因?yàn)?ArrayList 的底層數(shù)組,寫入值在數(shù)組滿的時候需要擴(kuò)容,在擴(kuò)容還沒完成的時候,新的下標(biāo)卻已經(jīng)計(jì)算出來并且要去插入,那么就會出現(xiàn)空指針異常。

有的讀有的寫。那么顯然對于多個線程來說,2 里面各個線程寫的情況對應(yīng)的問題就會出現(xiàn)。除此之外:

如果多線程有的讀有的寫,對于 ArrayList 底層,某些情況下,對象是不允許進(jìn)行修改的,如果修改了,后面調(diào)用某些方法時,就會檢測到,然后就直接拋出ConcurrentModificationException。 具體一下,因?yàn)樵创a里,寫操作對集合修改是寫,而next、remove等 Itr 的遍歷讀操作的時候會通過當(dāng)前集合的修改次數(shù)與 Itr 對象創(chuàng)建時記錄的次數(shù)校驗(yàn)集合是否被修改,如果修改了,不一致就說明正讀的時候還有別的線程在改,就會拋出異常。 JDK作者說了,會拋這個異常的都叫fail-fast iterator。

第 3 種情況就是對應(yīng)了我們上面的代碼在線程多起來的情況,因?yàn)檩敵?list 的時候需要遍歷的讀,而此時還有別的線程在進(jìn)行 add 的修改操作。

1.3 解決方法

注意:當(dāng)然不能自己加鎖,因?yàn)榧项愐呀?jīng)再演變過程有線程安全的替代品,自己的代碼加鎖的粒度已經(jīng)在集合的外層再加一層了,粒度太大。

同樣能夠完成 ArrayList 功能的,可以使用 Vector,查看源碼就會發(fā)現(xiàn),Vector 的基本結(jié)構(gòu)是一個叫 elementData 的 Object 類型的數(shù)組,和 ArrayList 類似,但是對應(yīng)的操作方法,基本都加上了 synchronized 關(guān)鍵字,因此它是線程安全的集合。 數(shù)據(jù)量小的時候,使用 Collections.synchronizedList(new ArrayList())這種方式,來包裹這個集合,跟 Collections 里面 synchronizedMap包裹hashmap 是一樣的,更多的,還有:

java的各種集合為什么不安全(List、Set、Map)以及代替方案

顯然能傳入?yún)?shù)的這些基本集合類都是線程不安全的。

第三種就是,直接使用 juc 包里面的,CopyOnWriteArrayList() 類,這個類就是并發(fā)包給我們提供的線程安全的列表類。1.4里介紹了這個集合。

1.4 CopyOnWriteArrayList

對于 CopyOnWriteArrayList 類,名字上就可以聽的出來,寫時復(fù)制的列表。

首先,按照前面的我們的分析,只要涉及了寫的操作,和讀或者寫搭配的多線程情況,就會出現(xiàn)問題,那么多線程同時讀卻不會出現(xiàn)問題,因此相比較于直接都加上 synchronized 的方式,他的思想就是:讀寫分離。這個思想在數(shù)據(jù)庫對于高并發(fā)的架構(gòu)層面也有一樣的設(shè)計(jì)。

這樣一來,對于這個 List 集合來說,分為不同操作的保證線程安全的策略,就能夠保證更好的性能。

寫的方法,我們首先可以看 add 方法源碼:

java的各種集合為什么不安全(List、Set、Map)以及代替方案

步驟很清楚,如果有了寫操作,需要加鎖:

加鎖 獲取到當(dāng)前的集合數(shù)組; 計(jì)算長度; 調(diào)用 Arrays.copyOf 方法進(jìn)行添加操作,每次只添加一個元素進(jìn)去; 修改引用,更新最新的集合; return true。 解鎖

其中的 lock 在源碼里就是一個:

java的各種集合為什么不安全(List、Set、Map)以及代替方案

可以看到是一個普通的 Object。

那么加鎖的時候就用 synchronized 對 Object 進(jìn)行加鎖,沒有采用 juc 的 ReetrantLock,注釋li也寫了,偏向于使用內(nèi)置的 monitor 也就是 synchronized 底層 monitor 鎖,這一點(diǎn)也充分說明了 synchronized 的性能更新使得源碼作者使用它。

這個方法是處理最直接的,其他對應(yīng)的寫操作:remove、set等等也是一樣的基礎(chǔ)流程。

我們再來看看讀操作 get 方法:

java的各種集合為什么不安全(List、Set、Map)以及代替方案

二、HashSet 的不安全

2.1 問題及原因

我們還是用 List 一樣的測試代碼;

public class TestSet { public static void main(String[] args) { HashSet<String> set = new HashSet<>(); for (int i = 0; i < 100; i++){ new Thread(()->{set.add(UUID.randomUUID().toString().substring(0,8));System.out.println(set); },String.valueOf(i)).start(); } }}

就會看到一樣的錯誤:

java的各種集合為什么不安全(List、Set、Map)以及代替方案

2.2 出現(xiàn)問題的原因

其實(shí)從出現(xiàn) ConcurrentModificationException 異常來看,我們可以猜測是和 List 類似的原因?qū)е碌漠惓!?/p>

可以看到,源碼里面,Set 的底層維護(hù)的是一個 HashMap 來實(shí)現(xiàn)。對于遍歷操作來說,都是一樣的使用了 fail-fast iterator 迭代器,因此會出現(xiàn)這個異常。

另外,因?yàn)?HashSet 的底層是 HashMap ,本質(zhì)上,對于每一個 key ,保證唯一,使用了一個 value 為 PRESENT 常量的鍵值對進(jìn)行存儲。

java的各種集合為什么不安全(List、Set、Map)以及代替方案

put 的過程也是調(diào)用 map 的 put 方法。

2.3 解決方案

List 有對應(yīng)的 Vector 可用,本來就是線程安全的集合,但是 Set 沒有; 數(shù)據(jù)量小的時候,使用 Collections.synchronizedSet(new HashSet<>()) 這種方式,來包裹這個集合,上面我們使用 List 的時候也有類似的方法; 同樣的,juc包為我們提供了新的線程安全集合 CopyOnWriteArraySet()。

2.4 CopyOnWriteArraySet

按照前面的思路,List 的對應(yīng)線程安全集合是在 List 集合的數(shù)組基礎(chǔ)上進(jìn)行加鎖的相關(guān)操作。

那么 Set 既然底層是 HashMap,對應(yīng)的線程安全集合就應(yīng)該是對 HashMap 的線程安全集合進(jìn)行加鎖,或者說直接用 ConcurrentHashMap 集合來實(shí)現(xiàn) CopyOnWriteArraySet 。

但事實(shí)上,源碼并不是這么做的。

從名字來看,和 ConcurrentHashMap 也沒有什么關(guān)系,而是類似 CopyOnWriteArrayList 的命名,說明是讀寫單獨(dú)處理,來讓他成為線程安全的集合,那為什么是 ArraySet 多一個 array 修飾語呢?

java的各種集合為什么不安全(List、Set、Map)以及代替方案

可以看到,他的思路沒有順延 util 包的 HashSet 的實(shí)現(xiàn)思路,而是直接使用了 CopyOnWriteArrayList 作為底層數(shù)據(jù)結(jié)構(gòu)。也就是說沒有利用 Map 的鍵值對映射的特性來保證 set 的唯一性,而是用一個數(shù)組為基底的列表來實(shí)現(xiàn)。(那顯然在去重方面就要做額外的操作了。)

然后每一個實(shí)現(xiàn)的方法都很簡單,基本是直接調(diào)用了 CopyOnWriteArrayList 的方法:

java的各種集合為什么不安全(List、Set、Map)以及代替方案

我們最擔(dān)心的可能 產(chǎn)生問題的 remove 和 add 方法,也是使用了 CopyOnWriteArrayList 的方法:

而保證 set 的不重復(fù)性質(zhì)的關(guān)鍵,顯然就在于 CopyOnWriteArrayList 的 addIfAbsent 方法,我們還是點(diǎn)進(jìn) CopyOnWriteArrayList 源碼看一看這個方法的實(shí)現(xiàn):

java的各種集合為什么不安全(List、Set、Map)以及代替方案

其中的 indexOfRange 方法:

java的各種集合為什么不安全(List、Set、Map)以及代替方案

可以看到,也是加了 Monitor 鎖來進(jìn)行的,整個過程是這樣的:

獲取本來的 set ,是一個數(shù)組,以快照形式返回當(dāng)前的數(shù)組; indexOfRange 方法通過遍歷查找查找元素出現(xiàn)位置,addIfAbsent方法完成不存在則加入,如果前一個為 false 后一個就不會執(zhí)行; 加鎖; current 再次獲取一次當(dāng)前的快照,因?yàn)橛锌赡艿谝淮闻袛嗟倪^程有了其他線程的插入或者修改操作,此時已經(jīng)不像等,就進(jìn)入分支進(jìn)行判斷是否存在; 否則就要加入這個元素,和 CopyOnWriteArrayList 添加元素的最后操作是一樣的; 解鎖。

總結(jié)一下就是,線程安全的 Set 集合完全利用了 CopyOnWriteArrayList 集合的方法,對應(yīng)的操作也是讀寫分別處理,寫時復(fù)制的策略,通過 jvm 層面的鎖來保證安全,那么保證不重復(fù)的方法就是遍歷進(jìn)行比較。

這樣看來,相比于基于 HashMap 的去重方法,效率肯定會降低,不過如果基于線程安全的 HashMap ,插入操作從hash、比較、到考慮擴(kuò)容各方面會因?yàn)榧渔i的過程更復(fù)雜,而對于一個不重復(fù)的 Set 來說,完全沒必要,所以應(yīng)該綜合考慮之下采用了 List 為基礎(chǔ),暴力循環(huán)去重。

三、HashMap 的線程不安全

關(guān)于 HashMap 的相關(guān)問題,源碼里已經(jīng)分析過,大體是這樣的。

不安全:

普通讀寫不一致問題; 死循環(huán)問題; ConcurrentModificationException 異常。

解決:

util包的Hashtable集合線程安全; 用 synchronizedMap(new HashMap())包裝; 使用 juc 包的 ConcurrentHashMap。

HashMap 和 ConcurrentHashMap 的源碼分析:

HashMap源碼解析、jdk7和8之后的區(qū)別、相關(guān)問題分析

ConcurrentHashMap源碼解析,多線程擴(kuò)容

到此這篇關(guān)于java的各種集合為什么不安全(List、Set、Map)以及代替方案的文章就介紹到這了,更多相關(guān)java 集合不安全內(nèi)容請搜索好吧啦網(wǎng)以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持好吧啦網(wǎng)!

標(biāo)簽: Java
相關(guān)文章:
主站蜘蛛池模板: 久久精品国产亚洲aa | 最新国产精品亚洲二区 | 欧美人成a视频www | 国产一级淫片a免费播放口之 | 国产深夜福利视频网站在线观看 | 亚洲综合色就色手机在线观看 | 好吊色综合网天天高清 | 三级全黄的视频 | 亚州成人| 国产在线精品成人一区二区三区 | 久久国产免费观看精品1 | 黄色三级网址 | 日韩美香港a一级毛片 | 纯欧美一级毛片免费 | 手机看成人片 | 日本欧美一级aaaaa毛片 | 亚洲一区二区在线成人 | 欧美日韩另类综合 | 亚洲精品影院久久久久久 | 一级做a爱片特黄在线观看 一级做a爱片特黄在线观看免费看 | 国产精品久久久久久影院 | 欧美一级欧美一级高清 | 伊人久久在线 | 一级片大全 | 国产欧美曰韩一区二区三区 | 午夜福利国产一级毛片 | 国内精品小视频在线 | 日本一级毛片高清免费观看视频 | 亚洲美女在线播放 | 欧美在线视频一区二区 | 一区二区三区中文国产亚洲 | 最新版天堂资源中文官网 | 国产精品久久久亚洲 | 国产黄色在线播放 | 成年人在线观看视频网站 | japanesehd国产在线无毒不卡 | 国内精品小视频在线 | 欧洲成人在线视频 | 一级特级毛片免费 | 美女美女大片黄a大片 | 免费一区二区三区四区五区 |