在Java中如何避免ConcurrentModificationException

ConcurrentModificationException发生在遍历集合时直接修改结构,解决方法包括:使用Iterator的remove()方法安全删除;采用CopyOnWriteArrayList或ConcurrentHashMap等并发集合;遍历时收集待操作元素,结束后统一处理;避免增强for循环中调用集合增删方法。

在Java中,ConcurrentModificationException 通常发生在使用迭代器遍历集合的同时,直接通过集合本身修改其结构(如添加、删除元素),而迭代器并不知情。这种异常常见于 ArrayListHashMap 等非线程安全的集合类。为了避免这个问题,可以采用以下几种方式。

使用Iterator的remove方法

在遍历集合时,如果需要删除元素,应使用迭代器提供的 remove() 方法,而不是集合本身的 remove() 方法。

  • 迭代器的 remove() 方法是安全的,它会同步更新迭代器内部的状态。
  • 不能在一次迭代中连续调用两次 remove(),否则会抛出异常。

示例:

Iterator iterator = list.iterator();
while (iterator.hasNext()) {
    String item = iterator.next();
    if ("toRemove".equals(item)) {
        iterator.remove(); // 安全
    }
}

使用支持并发的集合类

在多线程环境下,推荐使用 java.util.concurrent 包中的集合类,它们设计上就避免了此类问题。

  • CopyOnWriteArrayList:适用于读多写少的场景,每次修改都会创建新的副本,遍历时不会抛出异常。
  • ConcurrentHashMap:支持高效的并发访问,其迭代器是弱一致性的,不会抛出 ConcurrentModificationException

示例:

CopyOnWriteArrayList list = new CopyOnWriteArrayList<>
list.add("a"); list.add("b");
for (String s : list) {
    if ("a".equals(s)) {
        list.remove(s); // 允许,但不会影响当前迭代
    }
}

在遍历时收集要操作的元素,之后统一处理

如果不能使用并发集合或迭代器删除,可以在遍历时先记录需要删除或添加的元素,等遍历结束后再进行修改。

  • 适合复杂的条件判断和批量操作。
  • 避免在遍历中直接修改原集合。

示例:

List toRemove = new ArrayList<>();
for (String item : list) {
    if (needToRemove(item)) {
        toRemove.add(item);
    }
}
list.removeAll(toRemove);

使用增强for循环时特别注意

增强for循环(foreach)底层使用的是迭代器,因此在循环体内直接调用集合的 add()remove() 方法会触发异常。

  • 不要在 foreach 中修改集合结构。
  • 如有需要,改用普通迭代器方式或上述替代方案。

错误示例(会抛异常):

for (String item : list) {
    if ("bad".equals(item)) {
        list.remove(item); // 错误!
    }
}

基本上就这些。关键是在遍历集合时保持结构一致性,要么用安全的方法修改,要么延迟修改操作。理解集合的fail-fast机制有助于写出更健壮的代码。