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

博客讨论了在Java中遍历并修改集合时遇到的问题,特别是使用foreach循环时出现的`ConcurrentModificationException`。文章通过示例展示了如何从使用foreach循环改进为使用Iterator来避免这个问题,并解释了Iterator的工作原理和fail-fast机制。同时,建议在遍历集合时使用Iterator的`remove()`方法来安全地移除元素。
摘要由CSDN通过智能技术生成

来源:juejin.cn/post/6844903906449358856

业务中有需要过滤的需求,踩了 foreach 的坑。

本来是这样写的:

user.forEach(u -> {
   ageList.forEach(a -> {
        if (u.getId().equals(a)) {
            user.remove(u);
            }
        });
    });
}

改进后是这样的:

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() 就好了。

                                👇推荐大家关注一个公众号👇

Java,直接在`for`循环删除集合元素可能会导致`ConcurrentModificationException`异常,这是因为迭代器的快速失败行为。为了避免这种情况,推荐使用迭代器(Iterator)的`remove`方法来安全地删除元素。下面是使用迭代器删除集合元素的正确方式: ```java import java.util.ArrayList; import java.util.Iterator; public class RemoveElement { public static void main(String[] args) { ArrayList<Integer> list = new ArrayList<>(); list.add(1); list.add(2); list.add(3); list.add(4); Iterator<Integer> iterator = list.iterator(); while (iterator.hasNext()) { Integer element = iterator.next(); if (element == 3) { // 假设我们要删除元素3 iterator.remove(); // 使用迭代器的remove方法删除元素 } } System.out.println(list); // 输出修改后的集合 } } ``` 在这个例子,我们在遍历`ArrayList`时,使用`iterator()`方法获得一个迭代器,然后在`while`循环检查每个元素。如果元素符合条件(比如等于3),我们就调用迭代器的`remove()`方法来删除它。这样就不会抛出`ConcurrentModificationException`异常。 使用`for-each`循环来删除元素通常不是安全的做法,因为它没有提供直接调用`remove`的方法。但是,如果确实需要在`for-each`循环删除元素,可以在循环体内使用条件判断来调用迭代器的`remove`方法。例如: ```java for (Iterator<Integer> iterator = list.iterator(); iterator.hasNext(); ) { Integer element = iterator.next(); if (element == 3) { // 如果元素等于3,使用迭代器的remove方法删除 iterator.remove(); } else { // 继续迭代 iterator.next(); } } ``` 在`for-each`循环,我们不能直接在循环体内部使用`break`或`continue`语句来跳过迭代器的`next`方法调用,这样会导致`NoSuchElementException`异常。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值