答案:为避免ConcurrentModificationException,单线程中应使用Iterator的remove方法安全删除元素;多线程环境下推荐使用CopyOnWriteArrayList等并发集合;也可在遍历前收集待删除元素再统一移除;增强for循环中禁止直接修改集合,需改用Iterator或普通for循环。

在使用Java集合类进行迭代时,如果在遍历过程中对集合进行了修改(如添加或删除元素),就可能抛出ConcurrentModificationException。这个异常是为了防止在多线程或单线程环境下因并发修改导致的数据不一致问题而设计的“快速失败”(fail-fast)机制。下面介绍几种常见的处理方法。
使用Iterator的remove方法
在单线程环境中,如果需要在遍历过程中删除元素,应使用Iterator提供的remove()方法,而不是直接调用集合的remove()方法。
- Iterator的
remove()方法是安全的,它会更新内部的修改计数器,避免触发异常。 - 适用于
ArrayList、LinkedList等实现类。
示例:
Listlist = new ArrayList<>(Arrays.asList("a", "b", "c")); Iterator it = list.iterator(); while (it.hasNext()) { String item = it.next(); if ("b".equals(item)) { it.remove(); // 安全删除 } }
使用支持并发的集合类
在多线程环境下,推荐使用java.util.concurrent包中提供的线程安全集合,它们内部已处理了并发修改的问题。
立即学习“Java免费学习笔记(深入)”;
-
CopyOnWriteArrayList:适合读多写少的场景,每次修改都会创建新的副本,迭代时不会抛出异常。 -
ConcurrentHashMap:适用于高并发的Map操作,提供了更细粒度的锁控制。
示例:
Listlist = new CopyOnWriteArrayList<> (Arrays.asList("a", "b", "c")); for (String item : list) { if ("b".equals(item)) { list.remove(item); // 允许直接删除 } }
在遍历前收集要删除的元素
如果不想使用并发集合或Iterator,可以在遍历过程中先记录需要删除的元素,之后再统一删除。
- 适用于不能使用Iterator或并发集合的场景。
- 避免了边遍历边修改的问题。
示例:
Listlist = new ArrayList<>(Arrays.asList("a", "b", "c")); List toRemove = new ArrayList<>(); for (String item : list) { if ("b".equals(item)) { toRemove.add(item); } } list.removeAll(toRemove);
使用增强for循环时避免修改原集合
增强for循环(foreach)底层使用Iterator,因此在循环体内直接调用集合的add()或remove()方法会触发异常。
- 不要在foreach中直接修改集合。
- 改用普通for循环(从后往前删除)或Iterator方式。
错误示例:
for (String item : list) {
if ("b".equals(item)) {
list.remove(item); // 抛出ConcurrentModificationException
}
}
基本上就这些常见处理方式。选择哪种方法取决于你的使用场景:单线程下优先用Iterator,多线程考虑并发集合,或者通过暂存删除项来规避问题。关键是理解fail-fast机制的设计意图,合理规避即可。不复杂但容易忽略细节。










