溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務條款》

List集合多線程并發(fā)條件下不安全怎么辦

發(fā)布時間:2021-12-28 10:40:20 來源:億速云 閱讀:193 作者:小新 欄目:開發(fā)技術

這篇文章主要為大家展示了“List集合多線程并發(fā)條件下不安全怎么辦”,內(nèi)容簡而易懂,條理清晰,希望能夠幫助大家解決疑惑,下面讓小編帶領大家一起研究并學習一下“List集合多線程并發(fā)條件下不安全怎么辦”這篇文章吧。

前言

在日常開發(fā)過程中,List是我們常用的集合,比如查詢數(shù)據(jù)庫內(nèi)容返回值比會用一個集合來裝,但是在多線程并發(fā)的條件下,會出現(xiàn)安全問題嗎?下面我們就來測試一下,如果出現(xiàn)安全問題,該如何解決.

一、List集合使用模擬并發(fā)測試

1.1 單線程環(huán)境下

public static void main(String[] args) {
    // List集合
    List<String> list = new ArrayList<>();
    // 循環(huán)插入
    for (int i = 0; i < 10; i++) {
        list.add(UUID.randomUUID().toString().substring(0,5));
        System.out.println(list);
    }
}

List集合多線程并發(fā)條件下不安全怎么辦

可以看到單線程條件下,我們做list的插入操作完全沒問題,下面我們來模擬并發(fā)條件下執(zhí)行,會出現(xiàn)什么問題。

1.2 多線程環(huán)境下

public static void main(String[] args) {
    // List集合
    List<String> list = new ArrayList<>();
    // 循環(huán)插入
    for (int i = 0; i < 10; i++) {
        // 開啟線程執(zhí)行
        new Thread(()->{
            list.add(UUID.randomUUID().toString().substring(0,5));
            System.out.println(list);
        },"線程List").start();

    }
}

List集合多線程并發(fā)條件下不安全怎么辦

ArrayList在迭代的時候如果同時對其進行修改就會拋出java.util.ConcurrentModificationException異常,就是并發(fā)修改異常。

二、解決方案

2.1 使用Vector類

public static void main(String[] args) {
    // List集合
    List<String> list = new Vector<>();
    // 循環(huán)插入
    for (int i = 0; i < 10; i++) {
        // 開啟線程執(zhí)行
        new Thread(()->{
            list.add(UUID.randomUUID().toString().substring(0,5));
            System.out.println(list);
        },"線程List").start();

    }
}

Vector 是同步訪問的,它的add方法底層加了synchronized關鍵字修飾。

List集合多線程并發(fā)條件下不安全怎么辦

測試結果:

List集合多線程并發(fā)條件下不安全怎么辦

2.1 使用Collections.synchronizedList

public static void main(String[] args) {
   // List集合
    List<String> list = Collections.synchronizedList(new ArrayList<>());
    // 循環(huán)插入
    for (int i = 0; i < 10; i++) {
        // 開啟線程執(zhí)行
        new Thread(()->{
            list.add(UUID.randomUUID().toString().substring(0,5));
            System.out.println(list);
        },"線程List").start();

    }
}

查看底層源碼可以發(fā)現(xiàn)他也使用了synchronized關鍵字修飾。

List集合多線程并發(fā)條件下不安全怎么辦

2.3 使用并發(fā)容器CopyOnWriteArrayList

public static void main(String[] args) {
    // List集合
    List<String> list = new CopyOnWriteArrayList<>();
    // 循環(huán)插入
    for (int i = 0; i < 10; i++) {
        // 開啟線程執(zhí)行
        new Thread(()->{
            list.add(UUID.randomUUID().toString().substring(0,5));
            System.out.println(list);
        },"線程List").start();

    }
}

查看源碼它使用的是lock鎖機制。

List集合多線程并發(fā)條件下不安全怎么辦

寫入時復制,有多個線程調(diào)用的時候,寫入的時候,復制一份,避免覆蓋造成數(shù)據(jù)問題。就是在寫的時候不對原集合進行修改,而是重新復制一份,修改完之后,再移動指針。

從JDK1.5開始Java并發(fā)包里提供了兩個使用CopyOnWrite機制實現(xiàn)的并發(fā)容器,它們是CopyOnWriteArrayList和CopyOnWriteArraySet。CopyOnWrite容器非常有用,可以在非常多的并發(fā)場景中使用到。

解讀源碼:

/**
 * Appends the specified element to the end of this list.
 *
 * @param e element to be appended to this list
 * @return {@code true} (as specified by {@link Collection#add})
 */
public boolean add(E e) {
    final ReentrantLock lock = this.lock;//可重入鎖
    lock.lock();//加鎖
    try {
        Object[] elements = getArray();
        int len = elements.length;
        Object[] newElements = Arrays.copyOf(elements, len + 1);//拷貝新數(shù)組
        newElements[len] = e;
        setArray(newElements);//將引用指向新數(shù)組
        return true;
    } finally {
        lock.unlock();//解鎖
    }
}

add()在添加集合的時候加上了鎖,保證了同步,避免了多線程寫的時候會Copy出N個副本出來。

以上是“List集合多線程并發(fā)條件下不安全怎么辦”這篇文章的所有內(nèi)容,感謝各位的閱讀!相信大家都有了一定的了解,希望分享的內(nèi)容對大家有所幫助,如果還想學習更多知識,歡迎關注億速云行業(yè)資訊頻道!

向AI問一下細節(jié)

免責聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉載和分享為主,文章觀點不代表本網(wǎng)站立場,如果涉及侵權請聯(lián)系站長郵箱:is@yisu.com進行舉報,并提供相關證據(jù),一經(jīng)查實,將立刻刪除涉嫌侵權內(nèi)容。

AI