Java JUC中操作List安全類的集合案例
不安全的集合
在單線程應(yīng)用中,通常采取new ArrayList(),指定一個(gè)List集合,用于存放可重復(fù)的數(shù)據(jù)。
但在多線程下,往往會(huì)出現(xiàn)意想不到的問題,代碼如下所示:
import java.util.*; public class ListTest { public static void main(String[] args) throws InterruptedException { // 創(chuàng)建list集合 //List<String> lists = Arrays.asList("1", "2", "3"); // 不安全 List<String> lists = new ArrayList<>(); // 開啟十個(gè)線程增加數(shù)據(jù) for (int i = 1; i <= 40; i++) { new Thread(()->{ lists.add(UUID.randomUUID().toString().substring(0,5)); System.out.println(Thread.currentThread().getName()+"=="+lists); },String.valueOf(i)).start(); } } }
其運(yùn)行結(jié)果如下所示:
多線程操作同一集合對(duì)象信息,往往會(huì)出現(xiàn)java.util.ConcurrentModificationException異常報(bào)錯(cuò)信息。
Java中提供的安全措施
在java語言中,提供了一種新的List集合,java.util.Vector類,具體看下列代碼:
import java.util.*; public class ListTest { public static void main(String[] args) throws InterruptedException { // 創(chuàng)建list集合 //List<String> lists = Arrays.asList("1", "2", "3"); // 不安全 //List<String> lists = new ArrayList<>(); List<String> lists = new Vector<>(); // 開啟十個(gè)線程增加數(shù)據(jù) for (int i = 1; i <= 40; i++) { new Thread(()->{ lists.add(UUID.randomUUID().toString().substring(0,5)); System.out.println(Thread.currentThread().getName()+"=="+lists); },String.valueOf(i)).start(); } } }
運(yùn)行日志如下所示:
不會(huì)出現(xiàn)java.util.ConcurrentModificationException報(bào)錯(cuò)信息。
為什么能保證數(shù)據(jù)的安全操作?
采取了 synchronized 針對(duì)方法執(zhí)行調(diào)用者加鎖,保證add操作的多線程安全性!
JUC下的安全List集合
在JUC包下,提供有以下幾種創(chuàng)建安全集合的方式。
- 方式一:Collections.synchronizedList(new ArrayList<>());
import java.util.*; public class ListTest { public static void main(String[] args) throws InterruptedException { List<String> lists = Collections.synchronizedList(new ArrayList<>()); // 開啟十個(gè)線程增加數(shù)據(jù) for (int i = 1; i <= 40; i++) { new Thread(()->{ lists.add(UUID.randomUUID().toString().substring(0,5)); System.out.println(Thread.currentThread().getName()+"=="+lists); },String.valueOf(i)).start(); } } }
查看底層源碼實(shí)現(xiàn)邏輯
判斷傳入的 list 集合類型,判斷類型是否為 java.util.RandomAccess,如果是則采取java.util.Collections.SynchronizedRandomAccessList構(gòu)造集合,如果不是則采取java.util.Collections.SynchronizedList構(gòu)造集合。
源碼中對(duì)應(yīng)的add操作邏輯如下所示:
采取synchronized同步代碼塊的方式,對(duì)數(shù)據(jù)的add操作實(shí)現(xiàn)加鎖!
- 方式二:new CopyOnWriteArrayList();
import java.util.*; import java.util.concurrent.CopyOnWriteArrayList; public class ListTest { public static void main(String[] args) throws InterruptedException { List<String> lists = new CopyOnWriteArrayList<>(); // 開啟十個(gè)線程增加數(shù)據(jù) for (int i = 1; i <= 40; i++) { new Thread(()->{ lists.add(UUID.randomUUID().toString().substring(0,5)); System.out.println(Thread.currentThread().getName()+"=="+lists); },String.valueOf(i)).start(); } } }
源碼中的介紹如下:
顯而易見,其邏輯如下所示:
- 調(diào)用add方法后,拿到j(luò)ava.util.concurrent.locks.ReentrantLock對(duì)象信息。
- 調(diào)用 lock.lock() 拿到鎖!
- 將原數(shù)組對(duì)象copy操作,并創(chuàng)建原數(shù)組大小+1的新數(shù)組。
- 將新數(shù)據(jù)放入新數(shù)組中。
- 任何操作finally,都進(jìn)行鎖的釋放!
性能方面
JUC包下的Lock操作,都比synchronized性能更好!
到此這篇關(guān)于JUC中操作List安全類的集合案例的文章就介紹到這了,更多相關(guān)JUC中List安全類內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Java 實(shí)現(xiàn)FTP服務(wù)實(shí)例詳解
這篇文章主要介紹了Java 實(shí)現(xiàn)FTP服務(wù)實(shí)例詳解的相關(guān)資料,需要的朋友可以參考下2017-04-04MongoDB中ObjectId的誤區(qū)及引起的一系列問題
這篇文章主要介紹了MongoDB中ObjectId的誤區(qū)及引起的一系列問題,非常不錯(cuò),具有參考借鑒價(jià)值,需要的朋友可以參考下2016-12-12詳解基于Spring Boot與Spring Data JPA的多數(shù)據(jù)源配置
本篇文章主要介紹了詳解基于Spring Boot與Spring Data JPA的多數(shù)據(jù)源配置,非常具有實(shí)用價(jià)值,需要的朋友可以參考下2017-05-05使用ObjectMapper把Json轉(zhuǎn)換為復(fù)雜的實(shí)體類
這篇文章主要介紹了使用ObjectMapper把Json轉(zhuǎn)換為復(fù)雜的實(shí)體類操作,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2021-08-08關(guān)于springboot 配置文件中屬性變量引用方式@@解析
這篇文章主要介紹了關(guān)于springboot 配置文件中屬性變量引用方式@@解析,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過來看看吧2020-04-04java線程池ThreadPoolExecutor實(shí)現(xiàn)原理詳解
這篇文章主要介紹了java線程池ThreadPoolExecutor實(shí)現(xiàn)原理詳解,ThreadPoolExecutor是線程池實(shí)現(xiàn)類,會(huì)動(dòng)態(tài)創(chuàng)建多個(gè)線程,并發(fā)執(zhí)行提交的多個(gè)任務(wù),需要的朋友可以參考下2023-12-12