登录
首页 >  文章 >  java教程

循环中避免ArrayList索引偏移方法

时间:2026-04-16 21:16:39 351浏览 收藏

在遍历ArrayList时直接使用正向for循环调用remove()极易因数组元素前移导致索引偏移,从而跳过相邻元素——这是由ArrayList底层基于动态数组的实现机制决定的;本文系统剖析了问题根源,并推荐三种安全高效的解决方案:倒序遍历(逻辑清晰、性能优异)、Iterator.remove()(唯一被JDK明确保障的遍历中删除方式),以及Java 8+的removeIf()(声明式、简洁可靠),同时警示了增强for循环删除、正向循环调用remove(Object)等常见陷阱,帮你从根本上避开“删不干净”和并发修改异常的坑。

如何避免在循环中使用remove方法导致ArrayList索引偏移

为什么循环中调用 remove() 会跳过元素

因为 ArrayList 内部是数组实现,每次 remove(int index) 都会把该位置之后所有元素向前移动一位。如果正向 for 循环(i = 0; i )中删除第 i 个元素,下一轮 i 自增后指向的是原 i+1 位置的元素——而它已经被前移成了新 i 位置,等于被跳过了。

推荐做法:倒序遍历 + remove(int index)

从末尾开始删,前面的索引不受影响,逻辑最直白、性能好、无额外对象创建。

for (int i = list.size() - 1; i >= 0; i--) {
    if (shouldRemove(list.get(i))) {
        list.remove(i);
    }
}
  • 适用于已知要删哪些索引,或能用 get(i) 判断条件的场景
  • 注意:不要在循环体里调用 list.size() 做边界判断(虽安全但略低效),可提前缓存
  • 不能用增强 for 循环(for (E e : list))配合 remove(e),会抛 ConcurrentModificationException

更安全通用的方式:用 Iterator.remove()

这是唯一被明确允许在遍历时安全删除元素的迭代器方法,底层做了 modCount 同步处理。

Iterator<String> it = list.iterator();
while (it.hasNext()) {
    String s = it.next();
    if (s.startsWith("tmp")) {
        it.remove(); // ✅ 正确
    }
}
  • 必须用 it.remove(),而不是 list.remove(s),后者仍会触发并发修改异常
  • 每个 it.next() 后最多只能调用一次 it.remove(),重复调用会抛 IllegalStateException
  • 适合基于元素值或状态判断是否删除的场景

Java 8+ 的声明式替代:用 removeIf()

内部就是用 Iterator.remove() 实现的,语义清晰、代码简洁,推荐优先使用。

list.removeIf(s -> s == null || s.trim().isEmpty());
  • 参数是 Predicate,返回 true 的元素会被删除
  • 注意:该方法是结构性修改,会改变原集合;如需保留原列表,先 new ArrayList(original)
  • 不适用于需要在删除过程中做复杂副作用(比如记录日志、更新外部状态)的场景

倒序遍历和 Iterator.remove() 是最不容易出错的两种,但很多人会下意识写成正向 for + remove(Object)——这不仅有索引偏移问题,还可能误删第一个匹配项(因为 remove(Object) 删除的是首个 equals 匹配项,不是当前遍历到的那个)。真正要注意的,其实是「删谁」和「怎么删」这两层意图别混在一起。

终于介绍完啦!小伙伴们,这篇关于《循环中避免ArrayList索引偏移方法》的介绍应该让你收获多多了吧!欢迎大家收藏或分享给更多需要学习的朋友吧~golang学习网公众号也会发布文章相关知识,快来关注吧!

资料下载
相关阅读
更多>
最新阅读
更多>
课程推荐
更多>