0

0

深入理解Java Iterator.remove() 方法的工作原理与实践

花韻仙語

花韻仙語

发布时间:2025-11-22 20:38:02

|

776人浏览过

|

来源于php中文网

原创

深入理解Java Iterator.remove() 方法的工作原理与实践

java `iterator` 接口的 `remove()` 方法提供了一种安全且高效的方式,用于在遍历集合时移除元素。本文将深入探讨 `arraylist` 中 `iterator.remove()` 的内部实现原理,包括其如何利用 `lastret` 追踪元素索引、处理并发修改异常,并分析其时间复杂度,帮助开发者更好地理解和运用这一关键功能,从而避免常见的并发修改问题。

在Java集合框架中,Iterator 接口是遍历集合元素和安全移除元素的标准方式。尤其是在 ArrayList 等非线程安全的集合中,直接在循环中使用集合自身的 remove() 方法往往会导致 ConcurrentModificationException。Iterator.remove() 方法正是为了解决这一问题而设计的。

1. Iterator.remove() 的基本概念与使用

Iterator.remove() 方法允许开发者在迭代过程中从底层集合中移除当前迭代器指向的元素。它必须在调用 next() 方法之后,且在同一次 next() 调用和下一次 next() 调用之间只能被调用一次。

考虑以下示例代码,它展示了如何使用 Iterator.remove() 从 ArrayList 中移除偶数:

import java.util.ArrayList;
import java.util.Iterator;

public class IteratorRemoveExample {
    public static void main(String[] args) {
        ArrayList<Integer> list = new ArrayList<>();
        list.add(1);
        list.add(2);
        list.add(3);
        list.add(4);

        System.out.println("原始列表: " + list); // 输出: 原始列表: [1, 2, 3, 4]

        Iterator<Integer> it = list.iterator();

        while (it.hasNext()) {
            int x = it.next();
            if (x % 2 == 0) {
                it.remove(); // 移除偶数
            } else {
                System.out.print(x + " "); // 打印奇数
            }
        }
        // 预期输出: 1 3
        System.out.println("\n处理后的列表: " + list); // 输出: 处理后的列表: [1, 3]
    }
}

上述代码的输出是 1 3,并且 ArrayList 中只剩下 1 和 3。这表明 it.remove() 成功地从 ArrayList 中移除了偶数元素。

立即学习Java免费学习笔记(深入)”;

为什么不直接使用 list.remove(index) 或 list.remove(object)?

如果在上述 while 循环中,我们尝试使用 list.remove(x) 或 list.remove(someIndex),将会抛出 ConcurrentModificationException。这是因为 ArrayList 的迭代器实现了“快速失败”(fail-fast)机制。当迭代器创建后,如果集合在迭代器之外被修改(即 modCount 不等于 expectedModCount),迭代器会在下一次操作时抛出此异常,以防止不确定的行为。Iterator.remove() 是 Iterator 接口提供的一种“安全”的修改方式,它会更新 expectedModCount,从而避免 ConcurrentModificationException。

2. ArrayList 中 Iterator.remove() 的内部机制

为了理解 Iterator.remove() 如何工作,我们需要深入查看 ArrayList 内部迭代器 Itr 的实现。ArrayList 内部有一个名为 Itr 的私有内部类,它实现了 Iterator 接口。Itr 类中的 remove() 方法是实现关键逻辑的地方:

绘蛙
绘蛙

电商场景的AI创作平台,无需高薪聘请商拍和文案团队,使用绘蛙即可低成本、批量创作优质的商拍图、种草文案

下载
// 简化后的 ArrayList.Itr.remove() 源码
private class Itr implements Iterator<E> {
    int cursor;       // 下一个要返回的元素的索引
    int lastRet = -1; // 上一个由 next() 返回的元素的索引,如果未调用 next() 或已移除,则为 -1
    int expectedModCount = modCount; // 迭代器期望的修改次数

    public void remove() {
        if (lastRet < 0) {
            // 如果在 next() 之前调用 remove(),或连续调用 remove()
            throw new IllegalStateException();
        }
        checkForComodification(); // 检查是否有其他地方修改了集合

        try {
            // 调用 ArrayList 外部类的 remove 方法,移除 lastRet 索引处的元素
            ArrayList.this.remove(lastRet);
            // 移除后,cursor 指向的位置可能已经改变,需要更新
            cursor = lastRet;
            // 将 lastRet 重置为 -1,防止多次移除同一个元素
            lastRet = -1;
            // 更新 expectedModCount 以匹配当前的 modCount,防止后续操作抛出 ConcurrentModificationException
            expectedModCount = modCount;
        } catch (IndexOutOfBoundsException ex) {
            // 如果 ArrayList.this.remove 抛出异常,说明集合结构已损坏
            throw new ConcurrentModificationException();
        }
    }

    // ... 其他方法如 next(), hasNext(), checkForComodification()
}

关键字段和方法解释:

  • lastRet:这是一个非常重要的内部变量。它存储了上一次调用 next() 方法时返回的元素的索引。当 remove() 方法被调用时,它会使用 lastRet 来确定要移除哪个元素。如果 lastRet 为 -1,表示 next() 尚未被调用或该元素已被移除,此时调用 remove() 会抛出 IllegalStateException。
  • cursor:表示下一个将要由 next() 方法返回的元素的索引。当元素被移除后,cursor 会被更新为 lastRet 的值,因为移除操作会使后续元素前移。
  • modCount:这是 ArrayList 类的成员变量,记录了 ArrayList 结构被修改的次数(如添加、删除元素)。
  • expectedModCount:这是 Itr 类的成员变量,在迭代器创建时被初始化为当前的 modCount。每次迭代器执行 next() 或 remove() 操作时,都会先调用 checkForComodification() 来检查 modCount 是否与 expectedModCount 相等。如果不等,则说明集合在迭代器之外被修改,会抛出 ConcurrentModificationException。
  • ArrayList.this.remove(lastRet):这是实际执行元素移除操作的地方。它调用的是 ArrayList 外部类的 remove(int index) 方法。此方法会将 lastRet 索引处的元素移除,并将该索引之后的所有元素向左移动一位,以填补空缺。
  • lastRet = -1:在成功移除元素后,lastRet 被重置为 -1。这是为了确保在下次调用 next() 之前不能再次调用 remove()。
  • expectedModCount = modCount:在 remove() 操作成功完成并更新了 ArrayList 的 modCount 后,迭代器会将其自身的 expectedModCount 更新为新的 modCount。这样,迭代器就“知道”了这次修改,并允许后续的迭代操作继续进行。

3. 时间复杂度分析

Iterator.remove() 方法的时间复杂度主要取决于其内部调用的 ArrayList.this.remove(lastRet) 方法。

ArrayList 的 remove(int index) 方法在移除指定索引的元素后,需要将其后的所有元素向前移动一位。如果 ArrayList 中有 n 个元素,并且我们移除了索引为 lastRet 的元素,那么需要移动的元素数量大约是 n - 1 - lastRet。

  • 最坏情况:当 lastRet 为 0(即移除了第一个元素)时,需要移动 n-1 个元素,时间复杂度为 O(n)
  • 最好情况:当 lastRet 为 n-1(即移除了最后一个元素)时,不需要移动任何元素,时间复杂度为 O(1)
  • 平均情况:平均而言,lastRet 位于列表的中间,需要移动大约 n/2 个元素,时间复杂度为 O(n)

因此,对于 ArrayList 而言,Iterator.remove() 方法的平均时间复杂度是 O(n)。在循环中频繁调用 remove() 可能会导致性能问题,尤其是在列表较大时。

4. 注意事项与最佳实践

在使用 Iterator.remove() 时,需要牢记以下几点以避免常见的错误:

  1. 调用顺序:remove() 方法必须在调用 next() 方法之后调用。如果 next() 尚未被调用,或者在上次 next() 调用后已经调用过 remove(),再次调用 remove() 会抛出 IllegalStateException。
  2. 一次 next() 对应一次 remove():每次 next() 调用只能对应一次 remove() 调用。
  3. 避免并发修改:Iterator.remove() 是在单线程环境中安全修改集合的推荐方式。如果在迭代过程中,除了通过 Iterator.remove() 之外,还通过集合自身的 add()、remove() 或 clear() 等方法修改了集合,迭代器会抛出 ConcurrentModificationException。
  4. 线程安全集合:对于多线程环境,如果需要在迭代过程中进行修改,应考虑使用 java.util.concurrent 包下的线程安全集合,例如 CopyOnWriteArrayList。这些集合通常提供弱一致性迭代器,或者其修改操作本身就是线程安全的,不会抛出 ConcurrentModificationException。

示例:错误的使用方式

ArrayList<Integer> list = new ArrayList<>();
list.add(1); list.add(2); list.add(3);

Iterator<Integer> it = list.iterator();
it.remove(); // 错误!在 next() 之前调用,抛出 IllegalStateException
ArrayList<Integer> list = new ArrayList<>();
list.add(1); list.add(2); list.add(3);

Iterator<Integer> it = list.iterator();
while (it.hasNext()) {
    Integer x = it.next();
    if (x == 2) {
        list.remove(x); // 错误!通过集合自身修改,抛出 ConcurrentModificationException
    }
}

总结

Iterator.remove() 方法是 Java 集合框架中一个强大且重要的功能,它为开发者提供了一种在迭代过程中安全修改集合的机制。通过深入理解其内部工作原理,特别是 ArrayList 中 Itr 类的实现,以及 lastRet、cursor 和 modCount 等关键字段的作用,我们可以更好地掌握其使用方式,避免 IllegalStateException 和 ConcurrentModificationException 等常见问题。尽管其在 ArrayList 中的时间复杂度为 O(n),但在需要边遍历边删除的场景下,它仍然是首选的解决方案。正确地运用 Iterator.remove() 是编写健壮、高效 Java 代码的关键一环。

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

WorkBuddy
WorkBuddy

腾讯云推出的AI原生桌面智能体工作台

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
while的用法
while的用法

while的用法是“while 条件: 代码块”,条件是一个表达式,当条件为真时,执行代码块,然后再次判断条件是否为真,如果为真则继续执行代码块,直到条件为假为止。本专题为大家提供while相关的文章、下载、课程内容,供大家免费下载体验。

107

2023.09.25

string转int
string转int

在编程中,我们经常会遇到需要将字符串(str)转换为整数(int)的情况。这可能是因为我们需要对字符串进行数值计算,或者需要将用户输入的字符串转换为整数进行处理。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

1030

2023.08.02

int占多少字节
int占多少字节

int占4个字节,意味着一个int变量可以存储范围在-2,147,483,648到2,147,483,647之间的整数值,在某些情况下也可能是2个字节或8个字节,int是一种常用的数据类型,用于表示整数,需要根据具体情况选择合适的数据类型,以确保程序的正确性和性能。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

612

2024.08.29

c++怎么把double转成int
c++怎么把double转成int

本专题整合了 c++ double相关教程,阅读专题下面的文章了解更多详细内容。

334

2025.08.29

C++中int的含义
C++中int的含义

本专题整合了C++中int相关内容,阅读专题下面的文章了解更多详细内容。

235

2025.08.29

硬盘接口类型介绍
硬盘接口类型介绍

硬盘接口类型有IDE、SATA、SCSI、Fibre Channel、USB、eSATA、mSATA、PCIe等等。详细介绍:1、IDE接口是一种并行接口,主要用于连接硬盘和光驱等设备,它主要有两种类型:ATA和ATAPI,IDE接口已经逐渐被SATA接口;2、SATA接口是一种串行接口,相较于IDE接口,它具有更高的传输速度、更低的功耗和更小的体积;3、SCSI接口等等。

1926

2023.10.19

PHP接口编写教程
PHP接口编写教程

本专题整合了PHP接口编写教程,阅读专题下面的文章了解更多详细内容。

656

2025.10.17

php8.4实现接口限流的教程
php8.4实现接口限流的教程

PHP8.4本身不内置限流功能,需借助Redis(令牌桶)或Swoole(漏桶)实现;文件锁因I/O瓶颈、无跨机共享、秒级精度等缺陷不适用高并发场景。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

2395

2025.12.29

C# ASP.NET Core微服务架构与API网关实践
C# ASP.NET Core微服务架构与API网关实践

本专题围绕 C# 在现代后端架构中的微服务实践展开,系统讲解基于 ASP.NET Core 构建可扩展服务体系的核心方法。内容涵盖服务拆分策略、RESTful API 设计、服务间通信、API 网关统一入口管理以及服务治理机制。通过真实项目案例,帮助开发者掌握构建高可用微服务系统的关键技术,提高系统的可扩展性与维护效率。

76

2026.03.11

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
Kotlin 教程
Kotlin 教程

共23课时 | 4.3万人学习

C# 教程
C# 教程

共94课时 | 11.2万人学习

Java 教程
Java 教程

共578课时 | 81.1万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号