List如何才能一边遍历,一边删除

本文深入探讨了在Java中使用foreach循环直接从ArrayList中删除元素时遇到的java.util.ConcurrentModificationException异常原因,并提供了正确的删除方法,即使用Iterator的remove()方法。

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

这可能是很多新手第一时间想到的方法

public static void main(String[] args) {
    List<String> list = new ArrayList<>();
    list.add("baidu");
    list.add("google");

    for (String company : list) {
        if (company.equals("baidu")) {
            list.remove(company);
        }
    }

    System.out.println(list);
}

但是如果试着执行会发现,程序会抛java.util.ConcurrentModificationException异常(并发修改异常)。

为什么会抛java.util.ConcurrentModificationException异常?

foreach循环在实际执行时,其实使用的是Iterator,使用的核心方法是hasnext()next()。调用next()方法获取下一个元素时,第一行代码就是调用了checkForComodification();,而该方法的核心逻辑就是比较modCountexpectedModCount这2个变量的值。

刚开始modCountexpectedModCount的值相等,第1次获取元素是没问题的。但是当执行完remove操作的时候modCount的值就被修改了。再次checkForComodification的时候,就会java.util.ConcurrentModificationException异常。  

正确删除的方法

使用Iterator的remove()方法

public static void main(String[] args) {
    List<String> list = new ArrayList<>();
    list.add("baidu");
    list.add("google");

    Iterator<String> iterator = list.iterator();

    while (iterator.hasNext()) {
        String company = iterator.next();
        if (company.equals("baidu")) {
            iterator.remove();
        }
    }

    System.out.println(list);
}

为何iterator.remove()就不会报错?

iterator.remove()的时候,每次删除一个元素,都会将modCount的值重新赋值给expectedModCount,2个变量就相等了,这样就不会触发java.util.ConcurrentModificationException异常。

在 Python 中,不建议在循环中直接操作列表,因为这会导致删除元素后列表长度发生变化,可能会导致遍历出现问题。 但是,我们可以通过一些技巧实现一边遍历一边删除的效果。例如,可以使用 while 循环和 pop() 方法: ```python my_list = [1, 2, 3, 4, 5] i = 0 while i < len(my_list): if my_list[i] % 2 == 0: my_list.pop(i) else: i += 1 print(my_list) # [1, 3, 5] ``` 在上面的例子中,我们遍历列表 my_list 中的每个元素,并检查它是否为偶数。如果是,则使用 pop() 方法删除它;如果不是,则将索引 i 增加 1,继续遍历下一个元素。由于使用 pop() 方法删除元素后,后面的元素会自动向前移动,因此在下一次循环中,我们需要检查当前位置的元素,而不是下一个位置的元素。 需要注意的是,如果列表中有连续多个需要删除的元素,那么这个方法可能会漏掉一些元素。为了避免这种情况,我们可以使用另一种方法,即将需要删除的元素的索引记录下来,最后再删除它们: ```python my_list = [1, 2, 3, 4, 5] indexes_to_remove = [] for i in range(len(my_list)): if my_list[i] % 2 == 0: indexes_to_remove.append(i) for index in reversed(indexes_to_remove): my_list.pop(index) print(my_list) # [1, 3, 5] ``` 在上面的例子中,我们首先遍历列表 my_list 中的每个元素,并将需要删除的元素的索引记录在列表 indexes_to_remove 中。然后,我们使用 reversed() 函数将索引列表反转,从后往前依次删除元素。这样做可以确保删除元素后,后面的元素不会影响前面元素的索引。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值