还在 for 循环中 remove 元素?必须劝退……

本文分享了如何在Java中使用foreach遍历集合时避免ConcurrentModificationException,通过迭代器进行操作并举例改进前后的代码。重点在于理解Iterator的工作原理和并发修改的处理方法。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

来源:juejin.cn/post/6844903906449358856

业务中有需要过滤的需求,踩了 foreach 的坑。本来是这样写的:

user.forEach(u -> {   ageList.forEach(a -> {        if (u.getId().equals(a)) {            user.remove(u);            }        }) 河北红色教育培训 www.gsganxun.cn ;    });}

改进后是这样的:

Iterator<SocNearbyRespDto> ui = user.iterator();    while (ui.hasNext()){        SocNearbyRespDto u = ui.next();        ageList.forEach(a->{            if (!u.getId().equals(a)){                ui.remove();            }        });    }}

Java 中通常有三种循环

for (int i = 0; i < list.size(); i++) {    System.out.print(list.get(i) + ",");}Iterator iterator = list.iterator();while (iterator.hasNext()) {    System.out.print(iterator.next() + ",");}for (Integer i : list) {    System.out.print(i + ",");}

list.foreach 跟代码中第三种增强型 for 循环一样,反编译的内容是

    Integer i;    for(Iterator iterator = list.iterator(); iterator.hasNext(); System.out.println(i)){        i = (Integer)iterator.next();            }

这样在 for 循环中调用会出现 ConcurrentModificationException 异常。

Iterator是工作在一个独立的线程中,并且拥有一个 mutex 锁。Iterator被创建之后会建立一个指向原来对象的单链索引表,当原来的对象数量发生变化时,这个索引表的内容不会同步改变,所以当索引指针往后移动的时候就找不到要迭代的对象,所以按照 fail-fast 原则 Iterator 会马上抛出java.util.ConcurrentModificationException异常。

所以改成用 Iterator.remove() 就好了。

近期热文推荐:

1.1,000+ 道 Java面试题及答案整理(2022最新版)

2.劲爆!Java 协程要来了。。。

3.Spring Boot 2.x 教程,太全了!

4.别再写满屏的爆爆爆炸类了,试试装饰器模式,这才是优雅的方式!!

5.《Java开发手册(嵩山版)》最新发布,速速下载!

觉得不错,别忘了随手点赞+转发哦!

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值