Java导致ConcurrentModificationException所有原因

 更新时间:2026年01月20日 10:49:16   作者:猩火燎猿  
ConcurrentModificationException是Java集合框架抛出的运行时异常,表示集合在遍历过程中被结构性修改了,导致迭代器无法保证一致性,下面就来介绍一下几种原因,感兴趣的可以了解一下

1. 什么是 ConcurrentModificationException?

它是 Java 集合框架抛出的运行时异常,表示集合在遍历过程中被结构性修改了,导致迭代器无法保证一致性。

2. 异常触发的底层机制

Java 集合(如 ArrayList、HashSet 等)在创建迭代器时,会记录集合的结构性修改次数(modCount)。每次集合结构发生变化(如 add、remove),modCount 增加。迭代器内部有一个 expectedModCount,每次调用 next()/hasNext() 时,会检查 modCount 是否和 expectedModCount 一致。如果不一致,就抛出 ConcurrentModificationException。

3. 导致 ConcurrentModificationException 的所有常见原因

3.1 遍历过程中直接修改集合

错误代码示例:

List<String> list = new ArrayList<>(Arrays.asList("a", "b", "c"));
for (String s : list) {
    if (s.equals("b")) {
        list.remove(s); // 错误!遍历时直接修改集合
    }
}

原因: for-each 底层用的是 iterator,直接用 list.remove() 修改集合,modCount 改变,expectedModCount 没变,抛异常。

3.2 用 Iterator 遍历时,直接用集合的 add/remove 方法修改集合

Iterator<String> it = list.iterator();
while (it.hasNext()) {
    String s = it.next();
    if (s.equals("b")) {
        list.remove(s); // 错误!应使用 it.remove()
    }
}

3.3 多线程并发修改集合

一个线程遍历集合,另一个线程同时修改集合(结构性操作),也会抛异常。

示例:

List<Integer> list = new ArrayList<>(Arrays.asList(1,2,3));
new Thread(() -> {
    for (Integer i : list) {
        System.out.println(i);
    }
}).start();
 
new Thread(() -> {
    list.add(4); // 并发修改
}).start();

3.4 在 for-each 循环中调用集合的 remove/add

for (String s : list) {
    list.remove(s); // 错误
}

正确做法: 用 Iterator 的 remove 方法。

3.5 对 Map 的 keySet/values/entrySet 进行遍历时修改 Map

Map<String, Integer> map = new HashMap<>();
map.put("a", 1); map.put("b", 2);
for (String key : map.keySet()) {
    map.remove(key); // 错误
}

3.6 迭代器遍历时,集合结构被外部方法修改

如果在遍历过程中调用了会修改集合结构的方法(即使不是直接在循环体里),也会导致异常。

3.7 使用 fail-fast 集合(如 ArrayList、HashSet、HashMap)遍历时结构性修改

Java 的大多数集合都是 fail-fast 的(快速失败),即检测到并发修改就立即抛异常。

4. 什么是结构性修改?

结构性修改指的是影响集合元素数量或排列的操作,比如 add、remove、clear、put(Map),而仅仅修改元素内容(如 set(index, value))不算结构性修改。

5. 如何避免 ConcurrentModificationException?

用 Iterator 的 remove 方法删除元素

Iterator<String> it = list.iterator();
while (it.hasNext()) {
    String s = it.next();
    if (s.equals("b")) {
        it.remove();
    }
}

用并发安全集合
如 CopyOnWriteArrayListConcurrentHashMap,这些集合不会抛 ConcurrentModificationException。

遍历前收集要删除的元素,遍历后统一删除

List<String> toRemove = new ArrayList<>();
for (String s : list) {
    if (条件) toRemove.add(s);
}
list.removeAll(toRemove);

使用 ListIterator 的 add/remove/set 方法

6. 其他补充

  • ConcurrentModificationException 只是 fail-fast 机制的一部分,不能保证百分百检测到所有并发修改。
  • 在多线程环境下,优先使用并发集合或加锁处理。

7. 总结

所有原因本质:
遍历过程中集合结构被直接或间接修改,导致迭代器检测到不一致。

常见场景:

  • for-each 循环中直接 add/remove
  • Iterator 遍历时集合 add/remove
  • 多线程并发修改
  • Map 的 keySet/values/entrySet 迭代时修改 Map

避免方式:

  • 用 iterator.remove()
  • 用并发集合
  • 遍历前收集、后批量修改

8. 底层原理再深入

8.1 modCount 和 expectedModCount

  • 每个 fail-fast 集合(如 ArrayList、HashMap)内部有一个 modCount 字段,代表结构性修改次数。
  • 创建迭代器时,迭代器保存一份 expectedModCount
  • 每次迭代器操作(如 next()remove())时,都会检查 modCount 是否和 expectedModCount 一致。
  • 如果不一致,说明集合被外部修改,抛出 ConcurrentModificationException

源码片段(以 ArrayList 为例):

public E next() {
    checkForComodification(); // 检查
    ...
}
final void checkForComodification() {
    if (modCount != expectedModCount)
        throw new ConcurrentModificationException();
}

9. 特殊集合的处理

9.1 并发集合不会抛出该异常

  • CopyOnWriteArrayListConcurrentHashMap 等并发集合,内部机制不同,不会 fail-fast。
  • 例如,CopyOnWriteArrayList 每次修改都会复制一份新数据,迭代器遍历的是旧快照,不关心后续修改。

9.2 老的 Vector、Hashtable

  • 这些集合的方法都加了同步锁(synchronized),不会抛出 ConcurrentModificationException,但性能较低。

10. 实际项目中的规避策略

10.1 单线程环境

  • 删除元素用 iterator.remove()。
  • 遍历前收集需要删除的元素,遍历后统一删除。
  • 不要在 for-each 或普通 for 循环中直接 remove/add。

10.2 多线程环境

  • 使用并发集合(如 CopyOnWriteArrayList、ConcurrentHashMap)。
  • 使用同步块(synchronized)保护遍历和修改操作。
  • 分批处理:先收集需要修改的数据,后统一操作。

10.3 遍历 Map 的安全删除

  • 用 Iterator<Map.Entry<K,V>> 遍历,然后用 iterator.remove() 删除当前 entry。
Iterator<Map.Entry<String, Integer>> it = map.entrySet().iterator();
while (it.hasNext()) {
    Map.Entry<String, Integer> entry = it.next();
    if (entry.getValue() < 10) {
        it.remove();
    }
}

11. 排查和调试方法

11.1 查看异常堆栈

  • 异常堆栈会指向集合的迭代器方法(如 next()),分析调用链,定位哪里修改了集合。

11.2 检查所有集合修改点

  • 搜索代码中所有对集合的结构性操作(add、remove、clear等),看是否在遍历期间被调用。

11.3 多线程场景

  • 检查是否有线程并发修改集合,必要时加锁或用并发集合。

12. 面试延伸问题

  1. 什么是 fail-fast?什么是 fail-safe?举例说明。
    • fail-fast:检测到并发修改立即抛异常(如 ArrayList)。
    • fail-safe:迭代器遍历的是快照,不抛异常(如 CopyOnWriteArrayList)。
  2. 如何安全地在遍历过程中删除集合元素?
    • 用 iterator.remove()。
  3. ConcurrentModificationException 一定能检测到所有并发修改吗?
    • 不能,只能检测到部分典型场景。
  4. 为什么并发集合不会抛 ConcurrentModificationException?
    • 并发集合设计了特殊机制,如快照、分段锁等,保证遍历安全。

13. 真实案例分析

案例:批量删除数据库记录时同步维护缓存集合

假设你有一个缓存 List,批量删除数据库记录后也要同步删除 List 中的元素:

// 错误做法,可能抛异常
for (User user : cacheList) {
    if (shouldDelete(user)) {
        cacheList.remove(user);
    }
}
 
// 正确做法
Iterator<User> it = cacheList.iterator();
while (it.hasNext()) {
    User user = it.next();
    if (shouldDelete(user)) {
        it.remove();
    }
}

14. 代码示例:多线程并发修改

List<Integer> list = new ArrayList<>();
for (int i = 0; i < 100; i++) list.add(i);
 
Thread t1 = new Thread(() -> {
    for (int i : list) {
        // 遍历
    }
});
Thread t2 = new Thread(() -> {
    list.remove(50); // 并发修改
});
t1.start(); t2.start();
// 可能抛 ConcurrentModificationException

15. 总结

  • ConcurrentModificationException 是集合 fail-fast 机制的体现。
  • 本质原因是遍历期间结构性修改集合。
  • 规避方法:用 iterator.remove(),用并发集合,多线程加锁,遍历后统一修改。
  • 多线程场景优先用并发集合,普通集合加锁也可。
  • 面试常问底层原理、fail-fast 与 fail-safe、实际场景规避。

到此这篇关于Java导致ConcurrentModificationException所有原因的文章就介绍到这了,更多相关Java导致ConcurrentModificationException原因内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • springboot加载json配置全过程

    springboot加载json配置全过程

    Spring Boot加载JSON配置的步骤:实现PropertySourceLoader接口,创建spring.factories文件,并在resources下添加application.json,通过指定spring.profiles.active属性来激活配置文件
    2025-12-12
  • IDEA tomcat启动项目方式

    IDEA tomcat启动项目方式

    这篇文章主要介绍了IDEA tomcat启动项目方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教
    2025-03-03
  • 详解解Spring Boot高并发锁的使用方法

    详解解Spring Boot高并发锁的使用方法

    在高并发场景中,多个线程/用户会同时操作同一共享资源,如果不做控制,会导致数据错误,锁是解决这类问题的核心工具之一,下面就来介绍一下Spring Boot高并发锁的使用
    2025-08-08
  • Java HashSet集合存储遍历学生对象代码实例

    Java HashSet集合存储遍历学生对象代码实例

    这篇文章主要介绍了Java HashSet集合存储遍历学生对象代码实例,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下
    2020-04-04
  • 新手了解java基础知识(二)

    新手了解java基础知识(二)

    这篇文章主要介绍了Java基础知识,本文介绍了Java语言相关的基础知识、历史介绍、主要应用方向等内容,需要的朋友可以参考下,希望对你有所帮助
    2021-07-07
  • Docker和 Containerd 的区别解析

    Docker和 Containerd 的区别解析

    containerd 是一个来自 Docker 的高级容器运行时,并实现了 CRI 规范,它是从 Docker 项目中分离出来,之后 containerd 被捐赠给云原生计算基金会(CNCF)为容器社区提供创建新容器解决方案的基础,这篇文章主要介绍了Docker和 Containerd 的区别,需要的朋友可以参考下
    2024-03-03
  • springboot利用aspose预览office文件的实现过程

    springboot利用aspose预览office文件的实现过程

    这篇文章主要给大家介绍了关于springboot利用aspose预览office文件的相关资料,文中通过示例代码以及图文介绍的非常详细,对大家的学习或者工作具有一定的参考价值,需要的朋友可以参考下
    2021-06-06
  • MyBatis源码解析——获取SqlSessionFactory方式

    MyBatis源码解析——获取SqlSessionFactory方式

    这篇文章主要介绍了MyBatis源码解析——获取SqlSessionFactory方式,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2021-12-12
  • 二维码生成Java实现代码

    二维码生成Java实现代码

    这篇文章主要为大家详细介绍了二维码生成Java实现代码,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2017-05-05
  • 教你怎么用Java通过关键字修改pdf

    教你怎么用Java通过关键字修改pdf

    此方法只适合通过关键字位置,在pdf上添加字符直接上代码,代码比较长,大部分自己的理解都在代码注释中了,需要的朋友可以参考下
    2021-05-05

最新评论