17370845950

如何在Java中处理ConcurrentModificationException
ConcurrentModificationException发生在遍历集合时直接修改结构,正确做法是使用Iterator.remove()、并发集合如CopyOnWriteArrayList或延迟处理删除操作。

在Java中,ConcurrentModificationException 是一种常见的运行时异常,通常发生在使用迭代器遍历集合的同时,直接通过集合本身修改其结构(如添加、删除元素),而没有使用迭代器提供的安全方法。这种行为违反了“快速失败”(fail-fast)机制,从而触发异常。

理解ConcurrentModificationException的触发原因

大多数Java集合类(如 ArrayList、HashMap)在设计上采用“快速失败”策略。当一个线程正在遍历集合时,如果检测到集合的结构被意外修改(modCount 变化),就会抛出 ConcurrentModificationException。

常见错误示例:

List list = new ArrayList<>();
list.add("A");
list.add("B");

for (String s : list) {
    if (s.equals("A")) {
        list.remove(s); // ❌ 直接调用list.remove()会抛出异常
    }
}

使用Iterator的remove方法安全删除

在遍历过程中删除元素时,应使用 Iterator 提供的 remove() 方法,这是唯一允许在迭代期间修改集合结构的安全方式。

正确做法:

Iterator it = list.iterator();
while (it.hasNext()) {
    String s = it.next();
    if (s.equals("A")) {
        it.remove(); // ✅ 安全删除
    }
}

注意:调用 it.remove() 前必须先调用 it.next(),否则会抛出 IllegalStateException。

使用支持并发的集合类

如果需要在多线程环境下操作集合,或者允许多个修改与遍历同时进行,可以考虑使用 java.util.concurrent 包中的线程安全集合。

  • CopyOnWriteArrayList:适用于读多写少的场景。每次修改都会创建新的副本,遍历时不会抛出 ConcurrentModificationException。
  • ConcurrentHashMap:提供线程安全的 map 操作,支持高并发访问。

示例:

List list = new CopyOnWriteArrayList<>();
list.add("A");
list.add("B");

for (String s : list) {
    if (s.equals("A")) {
        list.remove(s); // ✅ 在CopyOnWriteArrayList中是安全的
    }
}

收集待操作元素,延迟处理

另一种思路是将需要删除或修改的元素先收集起来,等遍历结束后再统一处理。

示例:

List toRemove = new ArrayList<>
for (String s : list) {
    if (s.equals("A")) {
        toRemove.add(s);
    }
}
list.removeAll(toRemove); // ✅ 遍历结束后再删除

这种方式适合批量删除,逻辑清晰,但需要额外内存存储临时列表。

基本上就这些。关键是要避免在遍历过程中直接修改集合结构。根据具体场景选择使用迭代器、并发集合或延迟操作,就能有效避免 ConcurrentModificationException。不复杂但容易忽略。