為什么java不要在foreach循環里進行元素的remove/add操作
問題描述
選自《阿里巴巴JAVA開發手冊》
圖1代碼執行情況是:解釋刪除1這個元素不會報錯,但是刪除2這個元素報錯了,這個情況如何解釋?
問題解答
回答1:從報的錯誤中可以知道錯誤的來源 checkForComodification() ,如果要避免錯誤需要保持 modCount != expectedModCount 為 false 。list.remove(Object)會去調用fastRemove(int)方法,這個時候必然會去修改 modCount ,這個時候就會出現錯誤。Iterator<String> iterator = list.iterator() ;這個方法的實現就是返回一個內部類 Itr,(迭代的過程都是使用的這個類),但是為什么這個 iterator.remove() 不會出現錯誤了,原因在與這個方法的實現是在進行實際的 ArrayList.this.remove 之前進行的 checkForComodfication 檢查,remove 之后又使 expectedModCount = modCount,所以不會出現錯誤。
Itr.remove 的實現
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(); }}
如果有不對的地方請指出 @叉叉哥 @蒲柳隱逸
回答2:單線程的情況下,在遍歷List時刪除元素,必須要用Iterator的remove方法而不能使用List的remove方法,否則會ConcurrentModificationException。試想如果一個老師正在點整個班級所有學生的人數,而學生如果不遵守紀律一會出去一會進來,老師肯定點不下去。
多線程的情況下,參考我的一篇博客:http://xxgblog.com/2016/04/02...
回答3:首先,這涉及多線程操作,Iterator是不支持多線程操作的,List類會在內部維護一個modCount的變量,用來記錄修改次數舉例:ArrayList源碼
protected transient int modCount = 0;
每生成一個Iterator,Iterator就會記錄該modCount,每次調用next()方法就會將該記錄與外部類List的modCount進行對比,發現不相等就會拋出多線程編輯異常。
為什么這么做呢?我的理解是你創建了一個迭代器,該迭代器和要遍歷的集合的內容是緊耦合的,意思就是這個迭代器對應的集合內容就是當前的內容,我肯定不會希望在我冒泡排序的時候,還有線程在向我的集合里插入數據對吧?所以Java用了這種簡單的處理機制來禁止遍歷時修改集合。
至于為什么刪除“1”就可以呢,原因在于foreach和迭代器的hasNext()方法,foreach這個語法糖,實際上就是
while(itr.hasNext()){ itr.next()}
所以每次循環都會先執行hasNext(),那么看看ArrayList的hasNext()是怎么寫的:
public boolean hasNext() { return cursor != size;}
cursor是用于標記迭代器位置的變量,該變量由0開始,每次調用next執行+1操作,于是:你的代碼在執行刪除“1”后,size=1,cursor=1,此時hasNext()返回false,結束循環,因此你的迭代器并沒有調用next查找第二個元素,也就無從檢測modCount了,因此也不會出現多線程修改異常但當你刪除“2”時,迭代器調用了兩次next,此時size=1,cursor=2,hasNext()返回true,于是迭代器傻乎乎的就又去調用了一次next(),因此也引發了modCount不相等,拋出多線程修改的異常。
當你的集合有三個元素的時候,你就會神奇的發現,刪除“1”是會拋出異常的,但刪除“2”就沒有問題了,究其原因,和上面的程序執行順序是一致的。
回答4:因為你在對元素進行增刪的時候集合中的數量就改變了,那么在遍歷的時候就有可能會出現問題.比如一個集合有10個元素,那就應該要遍歷10次,當你對增加或刪除了一個元素,遍歷的次數就不對,所以會報錯
回答5:倒序刪除就可以了,反正list盡量不要remove。可以加delete標記
回答6:文檔中那個黃色的說明很有意思。
這個例子的執行結果會出乎大家的意料,那么試下把“1”換成“2”,會是同樣的結果嗎?
這個還是要看ArrayList的源碼,一看便知。
回答7:倒序刪除就可以了
回答8:ArrayList不是線程安全的,這樣相當于你在遍歷的時候修改了List。ArrayList在這種情況下是會拋出并發修改異常的。
相關文章:
1. Docker for Mac 創建的dnsmasq容器連不上/不工作的問題2. docker安裝后出現Cannot connect to the Docker daemon.3. css - 定位為absolute的父元素中的子元素 如何設置在父元素的下面?4. java - 請問在main方法中寫成對象名.屬性()并賦值,與直接參參數賦值輸錯誤是什么原因?5. java - Spring boot 讀取 放在 jar 包外的,log4j 配置文件,系統有創建日志文件,不寫入日志信息。6. mysql里的大表用mycat做水平拆分,是不是要先手動分好,再配置mycat7. java - socket類服務端如何防止被ddos攻擊?8. javascript - 圖片鏈接請求一直是pending狀態,導致頁面崩潰,怎么解決?9. python - beautifulsoup獲取網頁內容的問題10. 怎么用css截取字符?
