直接用增强for循环遍历并删除会抛ConcurrentModificationException;应使用Iterator.remove()、removeIf()、Stream.filter(),或多线程下选CopyOnWriteArrayList、ConcurrentHashMap或加锁。
rrentModificationException这是最常见的错误。Java集合(如ArrayList、HashMap)在迭代过程中,如果结构被修改(add/remove),其内部的modCount计数器与迭代器预期的expectedModCount不一致,就会触发快速失败机制,抛出ConcurrentModificationException。即使单线程下也如此,并非只发生在多线程场景。
这是最推荐的单线程方案。Iterator的remove()方法是唯一被设计为可在遍历时安全删除元素的方式,它会同步更新expectedModCount,避免异常。
next()之后立即调用remove(),否则抛IllegalStateException
next()最多对应一次remove(),不能重复调用Iteratorit = list.iterator(); while (it.hasNext()) { String s = it.next(); if (s.startsWith("A")) it.remove(); // 安全 }
若目标是“保留满足条件的元素”,比逐个判断删除更简洁高效。
Collection.removeIf(Predicate) 是JDK 8+内置方法,底层仍用Iterator,但封装了逻辑,语义清晰:list.removeIf(s -> s.startsWith("A"));Stream.filter()生成新集合,原集合不变,适合不可变语义或需保留原始数据的场景:Listfiltered = list.stream() .filter(s -> !s.startsWith("A")) .collect(Collectors.toList());
并发修改问题本质是竞态条件,需从数据结构或同步机制入手:
CopyOnWriteArrayList:读多写少场景,遍历时使用快照,写操作复制底层数组,无ConcurrentModificationException,但内存和性能开销大ConcurrentHashMap:支持安全的遍历与更新(如computeIfAbsent、replace),但不保证迭代过程看到最新修改(弱一致性)synchronized块包裹整个遍历+修改逻辑,简单直接,但会降低并发度虽能避开异常,但有明显缺陷:
list.toArray())后删原集合:逻辑割裂,易出错;若集合很大,浪费内存for(int i=list.size()-1; i>=0; i--)):仅适用于按索引删除,且无法处理List中重复元素的精确匹配逻辑,可读性和扩展性差