91超碰碰碰碰久久久久久综合_超碰av人澡人澡人澡人澡人掠_国产黄大片在线观看画质优化_txt小说免费全本

溫馨提示×

溫馨提示×

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

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

在foreach循環中JAVA集合不能添加或刪除元素的原因是什么

發布時間:2021-06-12 09:15:14 來源:億速云 閱讀:504 作者:小新 欄目:開發技術

這篇文章主要介紹在foreach循環中JAVA集合不能添加或刪除元素的原因是什么,文中介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們一定要看完!

1. 編碼強制規約

在《阿里巴巴Java開發手冊》中,針對集合操作,有一項規定,如下:

【強制】不要在 foreach 循環里進行元素的 remove/add 操作。remove 元素請使用 Iterator方式,如果并發操作,需要對 Iterator 對象加鎖。

public class SimpleTest {
    public static void main(String[] args) {
        List<String> list = Lists.newArrayList();
        list.add("1");
        list.add("2");
        list.add("3");
        list.add("4");
 
        //正例
        Iterator<String> iterator = list.iterator();
        while (iterator.hasNext()) {
            String item = iterator.next();
            if ("1".equalsIgnoreCase(item)) {
                iterator.remove();
            }
        }
 
        //反例
        for (String item : list) {
            if ("2".equals(item)) {
                list.remove(item);
            }
        }
    }
}

2. 原因分析

在循環或迭代時,會首先創建一個迭代實例,這個迭代實例的expectedModCount 賦值為集合的modCount.   

每當迭代器使? hashNext() / next() 遍歷下?個元素之前,都會檢測 modCount 變量與expectedModCount 值是否相等,相等的話就返回遍歷;否則就拋出異常【ConcurrentModificationException】,終?遍歷

如果在循環中添加或刪除元素,是直接調用集合的add,remove方法【導致了modCount增加或減少】,但這些方法不會修改迭代實例中的expectedModCount,導致在迭代實例中expectedModCount 與 modCount的值不相等,拋出ConcurrentModificationException異常

但迭代器中的remove,add方法,會在調用集合的remove,add方法后,將expectedModCount 重新賦值為modCount,所以在迭代器中增加、刪除元素是可以正常運行的。

可以參考ArrayList中的內部私有類Itr、ListItr的源碼

public Iterator<E> iterator() {
        return new Itr();
    }
 
/**
     * An optimized version of AbstractList.Itr
     */
    private class Itr implements Iterator<E> {
        int cursor;       // index of next element to return
        int lastRet = -1; // index of last element returned; -1 if no such
        int expectedModCount = modCount;
 
        Itr() {}
 
        //刪除了一些代碼
 
        public void remove() {
            if (lastRet < 0)
                throw new IllegalStateException();
            checkForComodification();
 
            try {
                ArrayList.this.remove(lastRet);
                cursor = lastRet;
                lastRet = -1;
                expectedModCount = modCount;
            } catch (IndexOutOfBoundsException ex) {
                throw new ConcurrentModificationException();
            }
        }
 
        final void checkForComodification() {
            if (modCount != expectedModCount)
                throw new ConcurrentModificationException();
        }
 
  }
   public E remove(int index) {
        rangeCheck(index);
 
        modCount++;
        E oldValue = elementData(index);
 
        int numMoved = size - index - 1;
        if (numMoved > 0)
            System.arraycopy(elementData, index+1, elementData, index,
                             numMoved);
        elementData[--size] = null; // clear to let GC do its work
 
        return oldValue;
    }

3. 相關知識介紹

3.1. 什么是快速失敗(fail-fast)?

快速失敗(fail-fast) 是 Java 集合的?種錯誤檢測機制。在使?迭代器對集合進?遍歷的時候,在多線程下操作?安全失敗(fail-safe)的集合類可能就會觸發 fail-fast 機制,導致拋出ConcurrentModificationException 異常。 

另外,在單線程下,如果在遍歷過程中對集合對象的內容進?了修改的話也會觸發 fail-fast 機制。

舉個例?:多線程下,如果線程 1 正在對集合進?遍歷,此時線程 2 對集合進?修改(增加、刪除、修改),或者線程 1 在遍歷過程中對集合進?修改,都會導致線程 1 拋出ConcurrentModificationException 異常。

3.2. 什么是安全失敗(fail-safe)呢?

采?安全失敗機制的集合容器,在遍歷時不是直接在集合內容上訪問的,?是先復制原有集合內容,在拷?的集合上進?遍歷。所以,在遍歷過程中對原集合所作的修改并不能被迭代器檢測到,故不會拋ConcurrentModificationException 異常。

以上是“在foreach循環中JAVA集合不能添加或刪除元素的原因是什么”這篇文章的所有內容,感謝各位的閱讀!希望分享的內容對大家有幫助,更多相關知識,歡迎關注億速云行業資訊頻道!

向AI問一下細節

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

AI

墨玉县| 泰和县| 民勤县| 富源县| 确山县| 大新县| 崇州市| 梅州市| 宝丰县| 赤城县| 禄丰县| 谷城县| 纳雍县| 图片| 抚松县| 合山市| 台东市| 泉州市| 西吉县| 双城市| 承德县| 新晃| 师宗县| 垫江县| 平江县| 休宁县| 安吉县| 鄂伦春自治旗| 无锡市| 精河县| 常德市| 佳木斯市| 女性| 五原县| 永胜县| 富阳市| 大荔县| 奉新县| 潮州市| 楚雄市| 鹿泉市|