concurrentmodificationexception由迭代时修改集合结构触发,java通过fail-fast机制检测此类并发修改。非线程安全集合如arraylist在单线程中遍历时直接调用list.remove()也会抛出该异常。解决方案包括:使用copyonwritearraylist或concurrenthashmap等线程安全集合;单线程中使用iterator的remove方法安全删除;多线程下通过synchronized或reentrantlock同步代码块。选择合适集合类型和操作方式可有效避免该异常。

在Java中,ConcurrentModificationException 通常出现在多线程环境下对集合进行迭代的同时修改其结构(如添加、删除元素)时。这个异常是“快速失败”(fail-fast)机制的一部分,用于检测并发修改行为,避免不可预料的结果。
理解ConcurrentModificationException的触发原因
大多数Java集合类(如 ArrayList、HashMap)不是线程安全的。当一个线程正在遍历集合时,另一个线程修改了集合的结构,迭代器就会抛出 ConcurrentModificationException。即使是单线程,如果在遍历过程中自己修改集合,也会触发该异常。
例如:
List<String> list = new ArrayList<>();
list.add("A"); list.add("B");
for (String s : list) {
if ("A".equals(s)) {
list.remove(s); // 抛出ConcurrentModificationException
}
}
使用线程安全的集合替代方案
为避免并发问题,可以使用 Java 提供的线程安全集合类:
立即学习“Java免费学习笔记(深入)”;
- CopyOnWriteArrayList:适用于读多写少的场景。每次修改都会创建新的副本,迭代时不会抛出异常。
- ConcurrentHashMap:支持高并发的 Map 实现,不会抛出 ConcurrentModificationException,且性能优于 synchronizedMap。
- Vector 和 Hashtable:老式同步集合,方法用 synchronized 修饰,性能较低,不推荐新项目使用。
示例:
List<String> list = new CopyOnWriteArrayList<>();
list.add("A"); list.add("B");
// 多线程或遍历时删除不会抛异常
for (String s : list) {
if ("A".equals(s)) {
list.remove(s); // 安全操作
}
}
使用迭代器的安全删除方式
如果仍使用非线程安全集合,在单线程中应使用 Iterator 的 remove 方法来安全删除元素:
Iterator<String> it = list.iterator();
while (it.hasNext()) {
String s = it.next();
if ("A".equals(s)) {
it.remove(); // 正确方式
}
}
注意:必须调用 it.remove(),而不是 list.remove(),否则仍会触发异常。
加锁控制访问(synchronized 或显式锁)
在多线程环境中,若需使用普通集合,可通过同步机制保护共享资源:
List<String> list = Collections.synchronizedList(new ArrayList<>());
synchronized(list) {
for (String s : list) {
// 执行操作
if (someCondition) {
list.remove(s);
}
}
}
或者使用 ReentrantLock 等显式锁实现更细粒度控制。
基本上就这些。关键是根据使用场景选择合适的集合类型和操作方式。读多写少用 CopyOnWriteArrayList,高并发映射用 ConcurrentHashMap,必要时加锁。避免在遍历时直接调用集合的修改方法,优先使用迭代器或线程安全版本。不复杂但容易忽略细节。










