ConcurrentModificationException由迭代时直接修改集合触发,Java通过fail-fast机制检测并发修改。使用Iterator.remove()、并发集合如CopyOnWriteArrayList、Stream API的removeIf或filter、加锁同步可避免该异常,核心是避免遍历时直接调用集合的结构修改方法。

在Java中,ConcurrentModificationException 是一种运行时异常,通常发生在使用迭代器遍历集合的同时,有其他线程或当前线程直接修改了集合的结构(如添加、删除元素),导致迭代器检测到不一致状态。这种异常常见于 ArrayList、HashMap 等非线程安全的集合类。
理解ConcurrentModificationException的触发机制
Java中的许多集合类(如ArrayList、LinkedList、HashMap)都实现了“快速失败”(fail-fast)机制。这意味着当一个线程正在通过迭代器遍历集合时,如果另一个线程修改了该集合的结构,迭代器会抛出 ConcurrentModificationException,以防止数据不一致或不可预测的行为。
以下代码将触发该异常:
Listlist = new ArrayList<>(); list.add("A"); list.add("B"); for (String item : list) { if ("A".equals(item)) { list.remove(item); // 直接修改集合,抛出ConcurrentModificationException } }
避免和处理并发修改异常的方法
要正确处理这类问题,关键是避免在遍历时直接修改原集合。以下是几种有效的解决方案:
立即学习“Java免费学习笔记(深入)”;
使用Iterator的remove方法Iterator接口提供了安全的 remove() 方法,可以在遍历过程中删除当前元素,而不会抛出异常。
Iterator使用支持并发的集合类iterator = list.iterator(); while (iterator.hasNext()) { String item = iterator.next(); if ("A".equals(item)) { iterator.remove(); // 安全删除 } }
在多线程环境下,推荐使用 java.util.concurrent 包下的线程安全集合,例如 CopyOnWriteArrayList 或 ConcurrentHashMap。
例如,CopyOnWriteArrayList 适用于读多写少的场景:
List使用Stream API进行过滤list = new CopyOnWriteArrayList<>(); list.add("A"); list.add("B"); for (String item : list) { if ("A".equals(item)) { list.remove(item); // 不会抛出ConcurrentModificationException } }
Java 8+ 提供了 Stream API,可以避免显式遍历和修改集合。
list.removeIf(item -> "A".equals(item)); // 简洁且安全
或者创建新的过滤后集合:
List加锁控制同步访问filtered = list.stream() .filter(s -> !"A".equals(s)) .collect(Collectors.toList());
在多线程环境中,若必须使用非线程安全集合,可通过 synchronized 块保证操作的原子性。
synchronized(list) {
Iterator it = list.iterator();
while (it.hasNext()) {
String item = it.next();
if (needRemove(item)) {
it.remove();
}
}
}
总结与最佳实践
ConcurrentModificationException 并不是程序崩溃的信号,而是设计上提醒开发者注意集合的并发访问问题。实际开发中应遵循以下建议:
- 遍历时如需删除元素,优先使用 Iterator.remove()
- 多线程环境下选用 CopyOnWriteArrayList、ConcurrentHashMap 等并发集合
- 利用 Java 8 的 removeIf 或 Stream 流式操作简化逻辑
- 避免在增强for循环中调用集合的 add/remove 方法
- 明确区分“结构修改”和“元素内容变更”——只有结构变化才会触发异常
基本上就这些。掌握这些方法后,就能有效避免并合理处理ConcurrentModificationException。










