关于ConcurrentModificationException:List如何删除元素

今天面试官突然问了我这个问题,让我陷入了深深的思考,虽然之前写代码也碰到过类似问题,但没有深究,这次来记录一下。

首先创建一个list

ArrayList<Integer> list = new ArrayList<>();
for (int i = 1; i <= 10; i++) {
    list.add(i);
}

1、首先使用普通的for循环删除元素

for (int i = 0; i < list.size(); i++) {
    if (i % 2 == 0) {
        list.remove(i);
    }
}
System.out.println(list);

输出结果是

[2, 3, 5, 6, 8, 9]

可以看到没有报错,这里的remove是根据位置删除,而不是根据值删除,再来仔细看一下为什么是这个结果,这段代码是删除偶数位置上的元素,首先一开始删除的是0位置上的1,但此时list后面的元素会向前移,也就是说此时i还是为0,但此时位置上的元素是2,然后删除2位置上的元素4,后面元素继续前移....以此类推,所以平时要注意这点。

2、使用增强for

for (int num : list) {
    if (num % 2 == 0) {
        list.remove(num);
    }
}

这里的remove是删除值,此时会报错,报错内容是

Exception in thread "main" java.util.ConcurrentModificationException
	at java.util.ArrayList$Itr.checkForComodification(ArrayList.java:911)
	at java.util.ArrayList$Itr.next(ArrayList.java:861)
	at Main.main(Main.java:13)

可以看到出现了ConcurrentModificationException这个错误,至于为什么,下面再看

3、使用迭代器

 Iterator<Integer> iterator = list.iterator();
while (iterator.hasNext()) {
    int num = iterator.next();
    if (num % 2 == 0) {
        list.remove(num);
    }
}

此时也会报ConcurrentModificationException错误,以上两种出错的原因都是一样的,ArrayList中的remove方法执行后,会将ArrayList中的modCount自增1,ArrayList中的内部类Iterator中也有remove方法,内部类中有expectedModCount字段,内部类中的remove方法先调用ArrayList中的remove方法,modCount自增1后,将modCount的值赋给expectedModCount,使二者相等,所以在调用next方法校验二者的大小时,不会发生报错。如果像上述代码中,使用了next来迭代但使用的收ArrayList中的remove方法,则会导致modCount加1而expectedModCount没有,两者不一致导致报错。

使用迭代器正确的删除方式为

Iterator<Integer> iterator = list.iterator();
    while (iterator.hasNext()) {
        int num = iterator.next();
        if (num % 2 == 0) {
            iterator.remove();
    }
}

此时输出为

[1, 3, 5, 7, 9]

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
ConcurrentModificationExceptionJava中的一个异常,通常在对集合进行遍历时发生。它表示在遍历过程中,集合的结构发生了修改,导致遍历操作无法继续进行。 这个异常的原因是在使用迭代器遍历集合时,同时进行了增删操作。例如,使用iterator.next()方法获取下一个元素后,又使用集合的add()或remove()方法对集合进行了修改。这种修改会导致迭代器的fail-fast机制触发,抛出ConcurrentModificationException异常。 解决这个问题的方法是使用迭代器的remove()方法来进行元素的删除操作,而不是使用集合的remove()方法。另外,也可以使用并发集合类,如ConcurrentHashMap或CopyOnWriteArrayList,它们提供了线程安全的迭代器。 总结来说,要避免出现ConcurrentModificationException异常,需要在遍历集合时避免对集合进行增删操作,或者使用线程安全的集合类。<span class="em">1</span><span class="em">2</span><span class="em">3</span> #### 引用[.reference_title] - *1* [java.util.ConcurrentModificationException: null 报错解决](https://blog.csdn.net/qq_51741039/article/details/126008727)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v93^chatsearchT3_2"}}] [.reference_item style="max-width: 33.333333333333336%"] - *2* [java.util.ConcurrentModificationException: null异常(已解决,原因出自于Java集合的快速失败机制)](https://blog.csdn.net/qq_45649807/article/details/126211551)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v93^chatsearchT3_2"}}] [.reference_item style="max-width: 33.333333333333336%"] - *3* [java.util.ConcurrentModificationException 解决方法](https://download.csdn.net/download/weixin_38723192/14093201)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v93^chatsearchT3_2"}}] [.reference_item style="max-width: 33.333333333333336%"] [ .reference_list ]

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值