ConcurrentModificationException的处理方式包括:1. 使用Iterator的remove方法安全删除元素;2. 采用CopyOnWriteArrayList等并发集合避免异常;3. 遍历时收集待操作元素,结束后统一处理;4. 利用Stream API进行无副作用的流式过滤。核心是避免遍历中直接修改原集合,根据场景选择Iterator、并发容器或延迟操作。

在Java中,ConcurrentModificationException 是一个常见的运行时异常,通常出现在遍历集合的同时对集合进行结构性修改(如添加、删除元素),而没有使用安全的方式。这个问题多发于使用 for-each 循环或 Iterator 遍历时。下面介绍几种有效的处理方式。
1. 使用 Iterator 的 remove 方法
当需要在遍历过程中删除元素时,应使用 Iterator 提供的 remove() 方法,这是线程不安全集合中最推荐的做法。
- 普通循环中直接调用集合的
remove()会触发异常 -
Iterator.remove()是唯一允许在遍历中安全删除元素的标准方法
示例:
Iteratoriterator = list.iterator(); while (iterator.hasNext()) { String item = iterator.next(); if ("toRemove".equals(item)) { iterator.remove(); // 安全删除 } }
2. 使用支持并发的集合类
如果多个线程同时操作集合,建议使用 java.util.concurrent 包中的线程安全集合。
立即学习“Java免费学习笔记(深入)”;
-
CopyOnWriteArrayList:适用于读多写少的场景,写操作会复制整个底层数组 -
ConcurrentHashMap:高效支持并发读写,不会抛出 ConcurrentModificationException
示例:
Listlist = new CopyOnWriteArrayList<>(); for (String item : list) { if ("delete".equals(item)) { list.remove(item); // 允许,因为是 CopyOnWriteArrayList } }
3. 遍历时收集待操作元素,后续统一处理
避免在遍历过程中直接修改原集合,可以先记录要删除或修改的元素,遍历结束后再操作。
这种方式逻辑清晰,适合复杂条件判断。
ListtoRemove = new ArrayList<>(); for (String item : list) { if (needRemove(item)) { toRemove.add(item); } } list.removeAll(toRemove);
4. 使用 Stream API(Java 8+)
通过流式处理,可以避免显式遍历和修改的冲突,代码更简洁且不易出错。
例如过滤出符合条件的元素生成新集合:
ListfilteredList = list.stream() .filter(item -> !"exclude".equals(item)) .collect(Collectors.toList());
原始集合不变,自然不会触发异常。
基本上就这些。关键在于理解“快速失败”(fail-fast)机制:大多数非同步集合在迭代时检测到结构变化就会抛出异常。选择合适的方法取决于使用场景——单线程下用 Iterator 或延迟操作,多线程则优先考虑并发集合。不复杂但容易忽略细节。










