欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

ArrayList源碼和多線程安全問題分析

 更新時間:2019年05月29日 11:34:31   作者:NULL  
這篇文章主要介紹了ArrayList源碼和多線程安全問題,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,下面小編和大家一起來學(xué)習(xí)一下吧

1.ArrayList源碼和多線程安全問題分析

在分析ArrayList線程安全問題之前,我們線對此類的源碼進(jìn)行分析,找出可能出現(xià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)多線程對此數(shù)據(jù)進(jìn)行操作的時候如果不進(jìn)行同步控制,即有可能會出現(xiàn)線程安全問題。

1.2 add方法可能出現(xiàn)的問題分析

首先我們看一下add的源碼如下:

public boolean add(E e) {
ensureCapacityInternal(size + 1);
elementData[size++] = e;
return true;
}

此方法中有兩個操作,一個是數(shù)組容量檢查,另外就是將元素放入數(shù)據(jù)中。我們先看第二個簡單的開始分析,當(dāng)多個線程執(zhí)行順序如下所示的時候,會出現(xiàn)最終數(shù)據(jù)元素個數(shù)小于期望值。

按照此順序執(zhí)行完之后,我們可以看到,elementData[n]的只被設(shè)置了兩次,第二個線程設(shè)置的值將前一個覆蓋,最后size=n+1。下面使用代碼進(jìn)行驗(yàn)證此問題。

1.3 代碼驗(yàn)證

首先先看下以下代碼,開啟1000個線程,同時調(diào)用ArrayList的add方法,每個線程向ArrayList中添加100個數(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é)果來看,最后輸出的結(jié)果會小于我們的期望值。即當(dāng)多線程調(diào)用add方法的時候會出現(xiàn)元素覆蓋的問題。

1.4 數(shù)組容量檢測的并發(fā)問題

在add方法源碼中,我們看到在每次添加元素之前都會有一次數(shù)組容量的檢測,add中調(diào)用此方法的源碼如下:

ensureCapacityInternal(size + 1);

容量檢測的相關(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);
}

容量檢測的流程圖如下所示:

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

第二個線程往數(shù)組中添加數(shù)據(jù)的時候由于數(shù)組容量為10,而此操作往index為10的位置設(shè)置元素值,因此會拋出數(shù)組越界異常。

1.5 代碼驗(yàn)證數(shù)組容量檢測的并發(fā)問題

使用如下代碼:

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方法后,我們可以看到控制臺輸出如下:

1.6 ArrayList中其他方法說明

ArrayList中其他包含對共享變量操作的方法同樣會有并發(fā)安全問題,只需要按照以上的分析方法分析即可。

以上就是本文的全部內(nèi)容,希望對大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。

相關(guān)文章

  • SpringBoot中的ImportSelector類動態(tài)加載bean詳解

    SpringBoot中的ImportSelector類動態(tài)加載bean詳解

    這篇文章主要介紹了SpringBoot中的ImportSelector類動態(tài)加載bean詳解,ImportSelector接口是spring中導(dǎo)入外部配置的核心接口,根據(jù)給定的條件(通常是一個或多個注釋屬性)判定要導(dǎo)入那個配置類,在spring自動化配置和@EnableXXX中都有它的存在,需要的朋友可以參考下
    2024-01-01
  • java取出list中某幾個屬性組成一個新集合的幾種方式

    java取出list中某幾個屬性組成一個新集合的幾種方式

    在Java開發(fā)中經(jīng)常需要對List中的對象進(jìn)行一些操作,例如對某個字段進(jìn)行過濾、排序等,這篇文章主要給大家介紹了關(guān)于java取出list中某幾個屬性組成一個新集合的幾種方式,需要的朋友可以參考下
    2024-03-03
  • Java文檔注釋超詳細(xì)講解

    Java文檔注釋超詳細(xì)講解

    這篇文章主要給大家介紹了關(guān)于Java文檔注釋的相關(guān)資料,文檔注釋主要是用來生成java開發(fā)文檔javadoc的,生成的開發(fā)文檔和Java本身的API幫助文檔是一樣的,需要的朋友可以參考下
    2023-10-10
  • Spring Security 在 Spring Boot 中的使用詳解【集中式】

    Spring Security 在 Spring Boot 中的使用詳解【集中式】

    這篇文章主要介紹了Spring Security 在 Spring Boot 中的使用【集中式】,本文通過圖文并茂的形式給大家介紹的非常詳細(xì),對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友可以參考下
    2020-10-10
  • IO密集型任務(wù)設(shè)置線程池線程數(shù)實(shí)現(xiàn)方式

    IO密集型任務(wù)設(shè)置線程池線程數(shù)實(shí)現(xiàn)方式

    這篇文章主要介紹了IO密集型任務(wù)設(shè)置線程池線程數(shù)實(shí)現(xiàn)方式,具有很好的參考價值,希望對大家有所幫助,如有錯誤或未考慮完全的地方,望不吝賜教
    2024-07-07
  • Java中的functor實(shí)現(xiàn)

    Java中的functor實(shí)現(xiàn)

    Java中的functor實(shí)現(xiàn)...
    2006-12-12
  • JAVA自定義異常使用方法實(shí)例詳解

    JAVA自定義異常使用方法實(shí)例詳解

    這篇文章主要介紹了JAVA自定義異常使用方法實(shí)例詳解,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友可以參考下
    2020-06-06
  • 詳解Java中使用泛型實(shí)現(xiàn)快速排序算法的方法

    詳解Java中使用泛型實(shí)現(xiàn)快速排序算法的方法

    這篇文章主要介紹了Java中使用泛型實(shí)現(xiàn)快速排序算法的方法,快速排序的平均時間復(fù)雜度為(n\log n),文中的方法立足于基礎(chǔ)而并沒有考慮優(yōu)化處理,需要的朋友可以參考下
    2016-05-05
  • springboot實(shí)現(xiàn)分頁功能的完整代碼

    springboot實(shí)現(xiàn)分頁功能的完整代碼

    Spring Boot是一個快速開發(fā)框架,它提供了很多便捷的功能,其中包括分頁查詢,下面這篇文章主要給大家介紹了關(guān)于springboot實(shí)現(xiàn)分頁功能的相關(guān)資料,文中通過實(shí)例代碼介紹的非常詳細(xì),需要的朋友可以參考下
    2023-04-04
  • Java9 Stream Collectors新增功能(小結(jié))

    Java9 Stream Collectors新增功能(小結(jié))

    這篇文章主要介紹了Java9 Stream Collectors新增功能(小結(jié)),文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2019-12-12

最新評論