遍历集合时直接调用remove()会触发ConcurrentModificationException;安全删除方式有四种:1. Iterator的remove()(最通用);2. removeIf()(Java 8+,简洁高效);3. 倒序for循环(仅List适用);4. 先收集后统一删除(灵活但耗内存)。

在遍历集合时直接调用 remove() 方法删除元素,极大概率会抛出 ConcurrentModificationException(并发修改异常)。这不是线程安全问题,而是 Java 集合的“快速失败”(fail-fast)机制在起作用——它通过 modCount 检查结构是否被非法修改。安全删除的关键,在于使用与遍历方式匹配的、受控的移除机制。
使用 Iterator 的 remove() 方法(推荐,适用于所有 Collection)
这是最通用、最安全的方式。Iterator 的 remove() 是唯一被设计为在遍历中安全删除元素的方法,它会同步更新内部状态,避免触发快速失败检查。
- 必须在调用
next()之后、且仅调用一次remove(),否则抛IllegalStateException - 不能用 for-each 循环(本质是隐式 Iterator),因为它不暴露 Iterator 引用
- 示例:
使用 removeIf()(Java 8+,简洁高效)
Collection.removeIf(Predicate) 内部已封装了安全的迭代逻辑,语义清晰,代码简洁,底层仍基于 Iterator 的 remove,但无需手动管理。
- 适合单条件批量删除,如按值、按属性、按规则过滤
- 不可用于需要在删除前执行复杂逻辑或获取被删元素的场景
- 示例:
倒序 for 循环(仅适用于 List,需索引访问)
对 ArrayList 等支持随机访问的 List,从后往前遍历时,删除当前索引元素不会影响尚未访问的前面元素的索引位置,从而规避越界和漏删问题。
立即学习“Java免费学习笔记(深入)”;
- 不适用于 LinkedList(随机访问低效)
- 注意循环条件:用
i >= 0,且每次删除后i不额外减 1(因为后续元素已前移) - 示例:
收集待删元素,遍历结束后统一删除(灵活但占内存)
先遍历原集合,将满足条件的元素(或其索引)存入临时容器,再调用 removeAll() 或逐个删除。适用于删除逻辑复杂、需多次判断或跨多个集合比对的场景。
- 注意:若存储的是元素引用,对 HashSet/TreeSet 等依赖
equals/hashCode的集合要确保正确实现 - 临时集合建议用 LinkedHashSet 保持插入顺序(如需按序处理)
- 示例:










