今天一朋友问了个问题,对于如下一段代码,运行后会有怎样的结果?

public class ArrayListTest {

    public static void main(String[] args) {
List<Integer> list = new ArrayList<>(Arrays.asList(1, 2, 3, 4, 5));
for (Integer num : list) {
if(num == 4){
list.remove(num);
}
}
System.out.println(list);
} }

由于本人之前对于集合框架中的fail-fast机制有过一些了解,所以看到这种写法,很自然的认为会出现java.util.ConcurrentModificationException异常,控制台并不会输出预期的结果。

但事实情况呢?看如下截图,执行成功了。

本人又尝试将if判断中num依次换成1,2,3,结果和预期一样了。

到底是什么原因呢,只有删除倒数第二个元素的场景结果和预期不一样。

本人带着好奇心、调试了一把源码,终于发现了庐山真面目。

类似于以上的forEach循环,其实是一种语法糖,调试后就会发现,list在循环时是由Iterator迭代器来实现的。

所以真实的代码则类似如下:

public class ArrayListTest {

    public static void main(String[] args) {
List<Integer> list = new ArrayList<>(Arrays.asList(1, 2, 3, 4, 5));
for (Iterator<Integer> itr = list.iterator(); itr.hasNext();){
int num = itr.next();
if(num == 4){
list.remove(num);
}
}
System.out.println(list);
} }

调试后终于发现了玄机,先看下面三个迭代器的关键属性:

cursor --迭代器的游标,元素的索引值,初始值为0

lastRet --返回最后一个元素的索引值、如果没有找到则返回-1

expectedModCount --修改次数的期望值,可以看到在迭代器初始化时,这个属性就被赋值为当前修改次数的值了。

checkForComodification --此方法用来检查修改次数是否发生变化,从而判断是否需要触发fail-fast异常。

在迭代过程中,每一次迭代,cursor会+1, 而itr.hasNext()会判断是否存在下一个元素、irt.next()获取下一个元素的值,最终直到不存在下一个元素,则迭代结束。

跟进源码发现,itr.hasNext()判断方法并不会调用checkForComodification方法来检查list在迭代中是否有被修改,只是判断游标和长度是否相等,不等时则认为存在下一个元素。

在删除倒数第二个元素的场景下:当倒数第二个元素时迭代完成,开始迭代最后一个元素时,此时cursor是4,size由于在迭代过程倒数第二个元素移除了,所以-1, 此时cursor和size相等,不会再进入下一个迭代,因此不会触发checkForComodification方法的fail-fast机制。

其他三种情况,再删除元素后的下一个迭代,由于会调用it.next()方法,则会触发ConcurrentModificationException异常。

通过这次对arraylist中iterator迭代器源码的深入学习,本人对于迭代器的运作机制又有了进一步的认知。后续还要再深入学习jdk中的集合框架源码,不能过于自信~

ArrayList之foreach循环删除倒数第二个元素,不触发fail-fast机制的更多相关文章

  1. ArrayList在foreach删除倒数第二个元素不抛并发修改异常的问题

    平时我们使用ArrayList比较多,但是我们是否知道ArrayList在进行foreach的时候不能直接通过list的add或者move方法进行删除呢, 原因就是在我们进行foreach遍历的时候, ...

  2. 关于java中ArrayList的快速失败机制的漏洞——使用迭代器循环时删除倒数第二个元素不会报错

    一.问题描述 话不多说,先上代码: public static void main(String[] args) throws InterruptedException { List<Strin ...

  3. Java循环删除集合多个元素的正确打开方式

    首先说下不正确的打开方式: 第一:使用for循环删除集合的元素,示例代码如下 ArrayList<String> list = new ArrayList<String>(Ar ...

  4. foreach循环里不能remove/add元素的原理

    foreach循环 ​    foreach循环(Foreach loop)是计算机编程语言中的一种控制流程语句,通常用来循环遍历数组或集合中的元素.Java语言从JDK 1.5.0开始引入forea ...

  5. 集合赋值及for循环删除符合条件的元素

    一.Java语言中ArrayList对象能直接赋值给另一个ArrayList对象吗? https://zhidao.baidu.com/question/399214655.html ArrayLis ...

  6. Python笔记:用for循环删除列表中的元素

    for运行过程中会有一个指针来记录当前循环的元素是哪一个,一开始这个指针指向第0个元素,然后获取它,接着删除第0个元素,这时候,原来是第1个的元素会变成第0个,当指针向后移动一次,指向了现在第1个元素 ...

  7. (CSDN 迁移) JAVA循环删除List的某个元素

    若列表中只可能存在一个则可以用简单的循环删除,不多说. 若列表中可能存在多个,尤其是可能有多个连续的需要删除,用简单循环有可能发生异常. 需要使用迭代器(Iterator),两种具体实现: 逻辑上是一 ...

  8. 循环删除List集合的元素

    之前在使用list集合循环删除元素的时候,竟然出现了集合内的元素不能删除成功的问题,之后整理了一下,发现大有玄机! 1.如果指定了list的size大小,会出现下标越界异常 List<Strin ...

  9. python循环删除list中的元素

    直接上例子: a = [1,2,3,4,5,6] for i in a: a.remove(i) print(a) 返回:[2, 4, 6] 循环a,想删除a的所有元素,但实际确有数据保留了下来,这是 ...

随机推荐

  1. ecshop怎么以幻灯版调用首页主广告显示

    今天在做一个商品网站我使用的是一套现成的ecshop模板了,这套模板下载下来多半是可以使用的,但是我发现第一屏的首页广告一直是不显示我想了很多办法都不显示了,后来一站长告诉我如何解决,下面我就来介绍e ...

  2. tcp内存占用/socket内存占用

    net.ipv4.tcp_mem 内核分配给TCP连接的内存,单位是Page,1 Page = 4096 Bytes,可用命令查看: #getconf PAGESIZE 4096 net.ipv4.t ...

  3. 基于TreeSoft实现异构数据同步

    一.为了解决数据同步汇聚,数据分发,数据转换,数据维护等需求,TreeSoft将复杂的网状的同步链路变成了星型数据链路.     TreeSoft作为中间传输载体负责连接各种数据源,为各种异构数据库之 ...

  4. 数据库的ACID和CAP

    传统数据库的ACID分别是:         A(Atomicty)原子性         B(Consistency)一致性         I(Isolation)独立性         D(Du ...

  5. [转帖]运维必读:Linux 的内存分页管理

    运维必读:Linux 的内存分页管理 https://cloud.tencent.com/developer/article/1356431 内存是计算机的主存储器.内存为进程开辟出进程空间,让进程在 ...

  6. 分布式自增ID算法snowflake

    分布式系统中,有一些需要使用全局唯一ID的场景,这种时候为了防止ID冲突可以使用36位的UUID,但是UUID有一些缺点,首先他相对比较长,另外UUID一般是无序的,作为索引非常不好,严重影响性能. ...

  7. 创建线程的三种方式(Thread、Runnable、Callable)

    方式一:继承Thread类实现多线程: 1. 在Java中负责实现线程功能的类是java.lang.Thread 类. 2. 可以通过创建 Thread的实例来创建新的线程. 3. 每个线程都是通过某 ...

  8. php中array_replace,array_splice和str_replace三个函数相互比较

    php中有一些功能相似或者是名称相似的函数,比如array_replace,array_splice和str_replace这三个函数,从名称来看前两个操作数组的,后一个操作字符串的. array_r ...

  9. BBS项目架构

    数据库设计 用户表(用的是auth_user那张表,通过自定义表继承AbstractUser) phone 电话 avatar 头像 create_time 创建时间#外键 blog 一对一个人站点表 ...

  10. WUSTOJ 1338: The minimum square sum(Java)

    题目链接:1338: The minimum square sum Description Given a prime p(p<108), you are to find min{x2+y2}, ...