17370845950

Java中如何处理集合操作中的ConcurrentModificationException
ConcurrentModificationException的处理方式包括:1. 使用Iterator的remove方法安全删除元素;2. 采用CopyOnWriteArrayList等并发集合避免异常;3. 遍历时收集待操作元素,结束后统一处理;4. 利用Stream API进行无副作用的流式过滤。核心是避免遍历中直接修改原集合,根据场景选择Iterator、并发容器或延迟操作。

在Java中,ConcurrentModificationException 是一个常见的运行时异常,通常出现在遍历集合的同时对集合进行结构性修改(如添加、删除元素),而没有使用安全的方式。这个问题多发于使用 for-each 循环或 Iterator 遍历时。下面介绍几种有效的处理方式。

1. 使用 Iterator 的 remove 方法

当需要在遍历过程中删除元素时,应使用 Iterator 提供的 remove() 方法,这是线程不安全集合中最推荐的做法。

  • 普通循环中直接调用集合的 remove() 会触发异常
  • Iterator.remove() 是唯一允许在遍历中安全删除元素的标准方法

示例:

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

2. 使用支持并发的集合类

如果多个线程同时操作集合,建议使用 java.util.concurrent 包中的线程安全集合。

  • CopyOnWriteArrayList:适用于读多写少的场景,写操作会复制整个底层数组
  • ConcurrentHashMap:高效支持并发读写,不会抛出 ConcurrentModificationException

示例:

List list = new CopyOnWriteArrayList<>();
for (String item : list) {
    if ("delete".equals(item)) {
        list.remove(item); // 允许,因为是 CopyOnWriteArrayList
    }
}

3. 遍历时收集待操作元素,后续统一处理

避免在遍历过程中直接修改原集合,可以先记录要删除或修改的元素,遍历结束后再操作。

这种方式逻辑清晰,适合复杂条件判断。

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

4. 使用 Stream API(Java 8+)

通过流式处理,可以避免显式遍历和修改的冲突,代码更简洁且不易出错。

例如过滤出符合条件的元素生成新集合:

List filteredList = list.stream()
    .filter(item -> !"exclude".equals(item))
    .collect(Collectors.toList());

原始集合不变,自然不会触发异常。

基本上就这些。关键在于理解“快速失败”(fail-fast)机制:大多数非同步集合在迭代时检测到结构变化就会抛出异常。选择合适的方法取决于使用场景——单线程下用 Iterator 或延迟操作,多线程则优先考虑并发集合。不复杂但容易忽略细节。