ConcurrentModificationException发生在遍历集合时直接修改结构,正确做法是使用Iterator.remove()、并发集合如CopyOnWriteArrayList或延迟处理删除操作。
在Java中,ConcurrentModificationException 是一种常见的运行时异常,通常发生在使用迭代器遍历集合的同时,直接通过集合本身修改其结构(如添加、删除元素),而没有使用迭代器提供的安全方法。这种行为违反了“快速失败”(fail-fast)机制,从而触发异常。
大多数Java集合类(如 ArrayList、HashMap)在设计上采用“快速失败”策略。当一个线程正在遍历集合时,如果检测到集合的结构被意外修改(modCount 变化),就会抛出 ConcurrentModificationException。
常见错误示例:
Listlist = new ArrayList<>(); list.add("A"); list.add("B"); for (String s : list) { if (s.equals("A")) { list.remove(s); // ❌ 直接调用list.remove()会抛出异常 } }
在遍历过程中删除元素时,应使用 Iterator 提供的 remove() 方法,这是唯一允许在迭代期间修改集合结构的安全方式。
正确做法:
Iteratorit = list.iterator(); while (it.hasNext()) { String s = it.next(); if (s.equals("A")) { it.remove(); // ✅ 安全删除 } }
注意:调用 it.remove() 前必须先调用 it.next(),否则会抛出 IllegalStateException。
如果需要在多线程环境下操作集合,或者允许多个修改与遍历同时进行,可以考虑使用 java.util.concurrent 包中的线程安全集合。
发访问。示例:
Listlist = new CopyOnWriteArrayList<>(); list.add("A"); list.add("B"); for (String s : list) { if (s.equals("A")) { list.remove(s); // ✅ 在CopyOnWriteArrayList中是安全的 } }
另一种思路是将需要删除或修改的元素先收集起来,等遍历结束后再统一处理。
示例:
ListtoRemove = new ArrayList<> for (String s : list) { if (s.equals("A")) { toRemove.add(s); } } list.removeAll(toRemove); // ✅ 遍历结束后再删除
这种方式适合批量删除,逻辑清晰,但需要额外内存存储临时列表。
基本上就这些。关键是要避免在遍历过程中直接修改集合结构。根据具体场景选择使用迭代器、并发集合或延迟操作,就能有效避免 ConcurrentModificationException。不复杂但容易忽略。