ArrayList源碼和多線程安全問(wèn)題分析
1.ArrayList源碼和多線程安全問(wèn)題分析
在分析ArrayList線程安全問(wèn)題之前,我們線對(duì)此類的源碼進(jìn)行分析,找出可能出現(xiàn)線程安全問(wèn)題的地方,然后代碼進(jìn)行驗(yàn)證和分析。
1.1 數(shù)據(jù)結(jié)構(gòu)
ArrayList內(nèi)部是使用數(shù)組保存元素的,數(shù)據(jù)定義如下:
transient Object[] elementData; // non-private to simplify nested class access
在ArrayList中此數(shù)組即是共享資源,當(dāng)多線程對(duì)此數(shù)據(jù)進(jìn)行操作的時(shí)候如果不進(jìn)行同步控制,即有可能會(huì)出現(xiàn)線程安全問(wèn)題。
1.2 add方法可能出現(xiàn)的問(wèn)題分析
首先我們看一下add的源碼如下:
public boolean add(E e) {
ensureCapacityInternal(size + 1);
elementData[size++] = e;
return true;
}
此方法中有兩個(gè)操作,一個(gè)是數(shù)組容量檢查,另外就是將元素放入數(shù)據(jù)中。我們先看第二個(gè)簡(jiǎn)單的開(kāi)始分析,當(dāng)多個(gè)線程執(zhí)行順序如下所示的時(shí)候,會(huì)出現(xiàn)最終數(shù)據(jù)元素個(gè)數(shù)小于期望值。

按照此順序執(zhí)行完之后,我們可以看到,elementData[n]的只被設(shè)置了兩次,第二個(gè)線程設(shè)置的值將前一個(gè)覆蓋,最后size=n+1。下面使用代碼進(jìn)行驗(yàn)證此問(wèn)題。
1.3 代碼驗(yàn)證
首先先看下以下代碼,開(kāi)啟1000個(gè)線程,同時(shí)調(diào)用ArrayList的add方法,每個(gè)線程向ArrayList中添加100個(gè)數(shù)字,如果程序正常執(zhí)行的情況下應(yīng)該是輸出:
list size is :10000
代碼如下:
private static List<Integer> list = new ArrayList<Integer>();
private static ExecutorService executorService = Executors.newFixedThreadPool(1000);
private static class IncreaseTask extends Thread{
@Override
public void run() {
System.out.println("ThreadId:" + Thread.currentThread().getId() + " start!");
for(int i =0; i < 100; i++){
list.add(i);
}
System.out.println("ThreadId:" + Thread.currentThread().getId() + " finished!");
}
}
public static void main(String[] args){
for(int i=0; i < 1000; i++){
executorService.submit(new IncreaseTask());
}
executorService.shutdown();
while (!executorService.isTerminated()){
try {
Thread.sleep(1000*10);
}catch (InterruptedException e){
e.printStackTrace();
}
}
System.out.println("All task finished!");
System.out.println("list size is :" + list.size());
}
當(dāng)執(zhí)行此main方法后,輸出如下:

從以上執(zhí)行結(jié)果來(lái)看,最后輸出的結(jié)果會(huì)小于我們的期望值。即當(dāng)多線程調(diào)用add方法的時(shí)候會(huì)出現(xiàn)元素覆蓋的問(wèn)題。
1.4 數(shù)組容量檢測(cè)的并發(fā)問(wèn)題
在add方法源碼中,我們看到在每次添加元素之前都會(huì)有一次數(shù)組容量的檢測(cè),add中調(diào)用此方法的源碼如下:
ensureCapacityInternal(size + 1);
容量檢測(cè)的相關(guān)源碼如下:
private void ensureCapacityInternal(int minCapacity) {
if (elementData == DEFAULTCAPACITY_EMPTY_ELEMENTDATA) {
minCapacity = Math.max(DEFAULT_CAPACITY, minCapacity);
}
ensureExplicitCapacity(minCapacity);
}
private void ensureExplicitCapacity(int minCapacity) {
modCount++;
// overflow-conscious code
if (minCapacity - elementData.length > 0)
grow(minCapacity);
}
容量檢測(cè)的流程圖如下所示:

我們以兩個(gè)線程執(zhí)行add操作來(lái)分析擴(kuò)充容量可能會(huì)出現(xiàn)的并發(fā)問(wèn)題:
當(dāng)我們新建一個(gè)ArrayList時(shí)候,此時(shí)內(nèi)部數(shù)組容器的容量為默認(rèn)容量10,當(dāng)我們用兩個(gè)線程同時(shí)添加第10個(gè)元素的時(shí)候,如果出現(xiàn)以下執(zhí)行順序,可能會(huì)拋出java.lang.ArrayIndexOutOfBoundsException異常。

第二個(gè)線程往數(shù)組中添加數(shù)據(jù)的時(shí)候由于數(shù)組容量為10,而此操作往index為10的位置設(shè)置元素值,因此會(huì)拋出數(shù)組越界異常。
1.5 代碼驗(yàn)證數(shù)組容量檢測(cè)的并發(fā)問(wèn)題
使用如下代碼:
private static List<Integer> list = new ArrayList<Integer>(3);
private static ExecutorService executorService = Executors.newFixedThreadPool(10000);
private static class IncreaseTask extends Thread{
@Override
public void run() {
System.out.println("ThreadId:" + Thread.currentThread().getId() + " start!");
for(int i =0; i < 1000000; i++){
list.add(i);
}
System.out.println("ThreadId:" + Thread.currentThread().getId() + " finished!");
}
}
public static void main(String[] args){
new IncreaseTask().start();
new IncreaseTask().start();
}
執(zhí)行main方法后,我們可以看到控制臺(tái)輸出如下:

1.6 ArrayList中其他方法說(shuō)明
ArrayList中其他包含對(duì)共享變量操作的方法同樣會(huì)有并發(fā)安全問(wèn)題,只需要按照以上的分析方法分析即可。
以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
Bean的自動(dòng)注入及循環(huán)依賴問(wèn)題
本文詳細(xì)介紹了Bean的自動(dòng)注入及循環(huán)依賴,文中通過(guò)代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)有一定的研究?jī)r(jià)值,感興趣的小伙伴可以閱讀參考2023-03-03
Java selenium上傳文件的實(shí)現(xiàn)
本文主要介紹了Java selenium上傳文件的實(shí)現(xiàn),文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2022-04-04
Java Poi 在Excel中輸出特殊符號(hào)的實(shí)現(xiàn)方法
這篇文章主要介紹了Java Poi 在Excel中輸出特殊符號(hào)的實(shí)現(xiàn)方法,本文通過(guò)實(shí)例代碼給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2020-07-07
spring boot使用WebClient調(diào)用HTTP服務(wù)代碼示例
這篇文章主要介紹了spring boot使用WebClient調(diào)用HTTP服務(wù)代碼示例,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2019-12-12
基于java實(shí)現(xiàn)簡(jiǎn)單的圖片類別識(shí)別
這篇文章主要為大家詳細(xì)介紹了如何基于java實(shí)現(xiàn)簡(jiǎn)單的圖片類別識(shí)別功能,文中的示例代碼講解詳細(xì),具有一定的借鑒價(jià)值,感興趣的小伙伴可以跟隨小編一起學(xué)習(xí)一下2023-12-12
往DAO類中注入@PersistenceContext和@Resource的區(qū)別詳解
這篇文章主要介紹了往DAO類中注入@PersistenceContext和@Resource的區(qū)別詳解,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2022-02-02
Spring?Cloud?整合?nacos實(shí)現(xiàn)動(dòng)態(tài)配置中心的詳細(xì)步驟
這篇文章主要介紹了Spring?Cloud?整合?nacos?實(shí)現(xiàn)動(dòng)態(tài)配置中心,整合步驟是通過(guò)添加依賴新建nacos配置,本文分步驟通過(guò)實(shí)例代碼給大家詳細(xì)講解,需要的朋友可以參考下2022-10-10
Spring Boot項(xiàng)目利用Redis實(shí)現(xiàn)session管理實(shí)例
本篇文章主要介紹了Spring Boot項(xiàng)目利用Redis實(shí)現(xiàn)session管理實(shí)例,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2017-06-06
Spring?加載?Application?Context五種方式小結(jié)
這篇文章主要介紹了Spring?加載?Application?Context五種方式小結(jié),具有很好的參考價(jià)值,希望對(duì)大家有所幫助。2022-01-01

