Java JUC中操作List安全類的集合案例
不安全的集合
在單線程應(yīng)用中,通常采取new ArrayList(),指定一個List集合,用于存放可重復(fù)的數(shù)據(jù)。
但在多線程下,往往會出現(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<>();
// 開啟十個線程增加數(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é)果如下所示:

多線程操作同一集合對象信息,往往會出現(xiàn)java.util.ConcurrentModificationException異常報錯信息。
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<>();
// 開啟十個線程增加數(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)行日志如下所示:

不會出現(xiàn)java.util.ConcurrentModificationException報錯信息。
為什么能保證數(shù)據(jù)的安全操作?

采取了 synchronized 針對方法執(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<>());
// 開啟十個線程增加數(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)造集合。
源碼中對應(yīng)的add操作邏輯如下所示:

采取synchronized同步代碼塊的方式,對數(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<>();
// 開啟十個線程增加數(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對象信息。
- 調(diào)用 lock.lock() 拿到鎖!
- 將原數(shù)組對象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)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Eclipse的Debug調(diào)試技巧大全(總結(jié))
這篇文章主要介紹了Eclipse的Debug調(diào)試技巧大全(總結(jié)),小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧2017-12-12
SpringBoot詳解如何整合Redis緩存驗(yàn)證碼
本文主要介紹了SpringBoot集成Redis實(shí)現(xiàn)驗(yàn)證碼的緩存簡單案例,文中通過示例代碼介紹的非常詳細(xì),具有一定的參考價值,感興趣的小伙伴們可以參考一下2022-07-07
SpringBoot項(xiàng)目中使用OkHttp獲取IP地址的示例代碼
OkHttp?是一個由?Square?開發(fā)的高效、現(xiàn)代的?HTTP?客戶端庫,用于?Android?和?Java?應(yīng)用程序,它支持?HTTP/2?和?SPDY?等現(xiàn)代網(wǎng)絡(luò)協(xié)議,并提供了多種功能和優(yōu)化,本文給大家介紹了SpringBoot項(xiàng)目中如何獲取IP地址,需要的朋友可以參考下2024-08-08
如何實(shí)現(xiàn)java執(zhí)行kettle并傳參數(shù)
文章主要介紹了在審批成功后如何使用Kettle傳遞批次號參數(shù)并執(zhí)行KTR文件,同時,提到了所需的主要POM依賴,并強(qiáng)調(diào)了個人經(jīng)驗(yàn)的價值,鼓勵大家參考和使用2025-01-01
IDEA?一直scanning?files?to?index的四種完美解決方法(VIP典藏版)
這篇文章主要介紹了IDEA?一直scanning?files?to?index的四種完美解決方法(VIP典藏版),推薦第四種方法,第四種方法摸索研究后得出,親測好用,需要的朋友參考下吧2023-10-10

