Java?ArrayList遍歷foreach與iterator時remove的區(qū)別
一、Iterator和foreach的區(qū)別
- 多態(tài)差別(foreach底層就是Iterator)
- Iterator是一個接口類型,他不關(guān)心集合或者數(shù)組的類型;
- for和foreach都需要先知道集合的類型,甚至是集合內(nèi)元素的類型;
1.為啥說foreach底層就是Iterator
編寫的代碼:

反編譯代碼:

二、foreach與iterator時remove的區(qū)別
先來看阿里java開發(fā)手冊
但1的時候不會報錯,2的時候就會報錯(java.util.ConcurrentModificationException)

首先來看一下ArrayList中iterator方法的實(shí)現(xiàn):

調(diào)用了new Itr(),生成Itr類(迭代器)。此時會給Itr的三個參數(shù)初始化。

- cursor代表下一次的索引位置(開始是0)
- size是集合的大小(2)
拋出異常類
next方法()的時候會檢查checkForComodification是否相等

modCount修改計(jì)數(shù)(每次add和remove都會+1)expectedModCount期望的最大計(jì)數(shù)
1.remove操作源碼分析
首先來看一下刪除“2”的情況:
第一次循環(huán):
因?yàn)榇藭r的modCount和expectedModCount都為2(因?yàn)閍dd了兩次所以modCount為2),所以第一次循環(huán)中不會拋出異常,拋出異常都是發(fā)生在不是第一次循環(huán)的情況中。在next方法走完后,foreach循環(huán)方法體中的remove方法的if條件判斷不滿足,就結(jié)束了本次循環(huán)。
第二次循環(huán):
第二次循環(huán)的hasNext和next方法都是能成功走完的,在這之后會進(jìn)入到foreach循環(huán)方法體中的remove方法中,進(jìn)行刪除元素。而此時的size-1變?yōu)榱?。在remove方法中的fastRemove方法中,會對modCount+1,也就變?yōu)榱?。

第三次循環(huán):
然后會走入到第三次循環(huán)中的hasNext方法中。按照正常的情況下該方法是會返回false的,但因?yàn)榇藭r的size已經(jīng)變?yōu)榱?,而此時的cursor為2(cursor代表下一次的索引位置),所以兩者不等,錯誤地返回了true,所以會繼續(xù)走入到next方法中的checkForComodification方法中,判斷此時的modCount和expectedModCount是否相等。因?yàn)榇藭r的modCount已經(jīng)變?yōu)榱?,和expectedModCount的值為2不等,所以在此拋出了ConcurrentModificationException異常。
再來看一下刪除“1”的時候?yàn)槭裁床粫伋霎惓#?/strong>
第一次循環(huán):
同上,此時的modCount和expectedModCount都為2,所以第一次循環(huán)中的hasNext和next方法都不會拋異常。在這之后會進(jìn)入到foreach循環(huán)方法體中的remove方法中,進(jìn)行刪除元素。同上,size-1變?yōu)榱?,而modCount+1變?yōu)榱?。
第二次循環(huán):
在第二次循環(huán)的hasNext方法中,此時的cursor為1,而size也是1,兩者相等。所以hasNext方法返回false,就跳出了foreach循環(huán),不會走到隨后的next方法中,也就不會拋出異常。
2.源碼步驟
第一次
第①句調(diào)用iterator(),

調(diào)用了new Itr(),生成Itr類(迭代器)。此時會給Itr的三個參數(shù)初始化。

此時expectedModCount == modCount == 2(因?yàn)閘ist調(diào)動了add方法,add方法會對modCount實(shí)現(xiàn)++操作)
第②句調(diào)用下面hasNext()方法,返回下一個要訪問元素的下標(biāo)cursor,因?yàn)槭堑谝淮窝h(huán),所以cursor為0,size為2 (0 != 2 true)
第③句調(diào)用next()方法,foreach循環(huán)方法體中的remove方法的if條件判斷不滿足,就結(jié)束了本次循環(huán)
第二次
第②句調(diào)用下面hasNext()方法,返回下一個要訪問元素的下標(biāo)cursor,第二次循環(huán),所以cursor為1,
size還是為2 (1 != 2 true)
第③句調(diào)用next()方法,正常取值,取到第一個元素"2";
第④句調(diào)用remove()方法,成功給list刪除元素。注意,在調(diào)用remove方法的時候,有modCount++。所有此時,modCount3,expectedModCount2,size1
第三次
第②句調(diào)用下面hasNext()方法,返回下一個要訪問元素的下標(biāo)cursor,第二次循環(huán),所以cursor為2,size為1
第③句調(diào)用next()方法,注意,在next()方法中第一句話就是調(diào)用checkForComodification();由于modCount(3) != expectedModCount(2),所以就拋了異常。
3.為啥都是底層都是iterator,為啥foreach會報錯

當(dāng)循環(huán)結(jié)束的時候,while (iterator.hasNext()) 會檢查是否有下個元素存在,在remove刪除2完成后,下次進(jìn)入cursor還是1,size也是1.
foreach的話,刪除remove2之后,下次進(jìn)入cursor是2,size是1,所以返回false,要走next方法,然后,進(jìn)行檢查,modCount=3,而expectedModCount=2


三、查看源碼方法
如果查看iterator下的ArrayList

remove也是如此
到此這篇關(guān)于Java ArrayList遍歷foreach與iterator時remove的區(qū)別的文章就介紹到這了,更多相關(guān)Java ArrayList遍歷內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
springboot項(xiàng)目配置logback日志系統(tǒng)的實(shí)現(xiàn)
這篇文章主要介紹了springboot項(xiàng)目配置logback日志系統(tǒng)的實(shí)現(xiàn),文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2021-04-04
java實(shí)現(xiàn)簡易版圖形界面計(jì)算器
這篇文章主要為大家詳細(xì)介紹了java實(shí)現(xiàn)簡易版圖形界面計(jì)算器,文中示例代碼介紹的非常詳細(xì),具有一定的參考價值,感興趣的小伙伴們可以參考一下2020-05-05
Java編程實(shí)現(xiàn)向文本文件中讀取數(shù)據(jù)之Scanner用法示例
這篇文章主要介紹了Java編程實(shí)現(xiàn)向文本文件中讀取數(shù)據(jù)之Scanner用法,結(jié)合實(shí)例形式分析了java使用Scanner類讀取文本文件相關(guān)操作技巧與注意事項(xiàng),需要的朋友可以參考下2018-03-03
數(shù)組重排序(如何將所有奇數(shù)都放在所有偶數(shù)前面)的深入分析
本篇文章是對數(shù)組重排序(如何將所有奇數(shù)都放在所有偶數(shù)前面)的方法進(jìn)行了詳細(xì)的分析介紹,需要的朋友參考下2013-06-06
SpringWebMVC的常用注解及應(yīng)用分層架構(gòu)詳解
這篇文章主要介紹了SpringWebMVC的常用注解及應(yīng)用分層架構(gòu),SpringWebMVC是基于ServletAPI構(gòu)建的原始Web框架,從?開始就包含在Spring框架中,感興趣的朋友可以參考下2024-05-05
springboot發(fā)送request請求的方式小結(jié)
在Java中,發(fā)送HTTP請求是常見需求,hutool工具包和RestTemplate類是實(shí)現(xiàn)此功能的兩種主流方法,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2024-09-09
Java實(shí)現(xiàn)簡單酒店管理系統(tǒng)
這篇文章主要為大家詳細(xì)介紹了Java實(shí)現(xiàn)簡單酒店管理系統(tǒng),文中示例代碼介紹的非常詳細(xì),具有一定的參考價值,感興趣的小伙伴們可以參考一下2022-05-05
springboot接入方式對接股票數(shù)據(jù)源API接口的操作方法
本文介紹了如何使用Java語言創(chuàng)建一個項(xiàng)目來對接StockTV的API接口,包括使用HttpURLConnection或OkHttp發(fā)送HTTP請求,使用Java-WebSocket庫處理WebSocket連接等步驟,項(xiàng)目結(jié)構(gòu)包括添加依賴、創(chuàng)建基礎(chǔ)工具類、實(shí)現(xiàn)股票API、外匯API等,感興趣的朋友一起看看吧2025-03-03

