当前位置: 首页 > news >正文

一行一行读Java源码——迭代器

迭代器

我们都知道,当我们需要删除List中元素时,必须使用迭代器来操作,为什么需要使用迭代器来进行remove操作,而不能在for循环中删除?那么迭代器又是什么呢?

迭代器接口

以下代码是一个基本的迭代器接口,声明了迭代器的基本方法,而我们常用的就是hasNext、next、remove

package java.util;

import java.util.function.Consumer;

public interface Iterator<E> {
    
    boolean hasNext();

    E next();

    default void remove() {
        throw new UnsupportedOperationException("remove");
    }

    // Java 1.8加入,用于支持lambda表达式
    default void forEachRemaining(Consumer<? super E> action) {
        Objects.requireNonNull(action);
        while (hasNext())
            action.accept(next());
    }
}

每一种数据结构,其迭代器的实现必然存在差异,此处我以List为例。
List统称为线性表,而线性表又分为顺序表和链表。接下来,我们来看看AbstractList中是如何实现这两种类型迭代器的。(顺序表也就是常见的数组)

顺序表的迭代器实现——Itr

private class Itr implements Iterator<E> {
        // 游标,指示将要访问的元素的位置
        int cursor = 0;

        // 指示最后一次访问位置,如果是最后一次是删除操作,该值为-1
        int lastRet = -1;

        // 同步校验位,主要用于remove时校验是否有多线程并行删除元素,此机制类似乐观锁
        int expectedModCount = modCount;

        // 是否还有可访问元素
        public boolean hasNext() {
            return cursor != size();
        }
        
        // 校验顺序表是否被其它线程修改过,未修改才可访问,否则抛ConcurrentModificationException异常
        // 返回游标指示元素
        // 标记最近一次访问位置为游标位置
        // 游标向前推进一位
        public E next() {
            checkForComodification();
            try {
                int i = cursor;
                E next = get(i);
                lastRet = i;
                cursor = i + 1;
                return next;
            } catch (IndexOutOfBoundsException e) {
                checkForComodification();
                throw new NoSuchElementException();
            }
        }

        // 判断该位置元素是否已经删除过
        // 并发访问校验
        // 调用List的当前实例删除当前(最近访问)元素
        // 游标回退一位
        // 最近访问的元素被删除后,那最近访问的元素下标标记为-1
        // 因为AbstractList.this.remove(lastRet)中的remove方法会修改modCount的值,所有expectedModCount也需要被重新赋值
        public void remove() {
            if (lastRet < 0)
                throw new IllegalStateException();
            checkForComodification();

            try {
                AbstractList.this.remove(lastRet);
                if (lastRet < cursor)
                    cursor--;
                lastRet = -1;
                expectedModCount = modCount;
            } catch (IndexOutOfBoundsException e) {
                throw new ConcurrentModificationException();
            }
        }

        // 校验当前List的修改数modCount是否和迭代器中存储的一致,防止多线程并发访问
        final void checkForComodification() {
            if (modCount != expectedModCount)
                throw new ConcurrentModificationException();
        }
    }

链表的迭代器实现——ListItr

// 链表的迭代器实现继承了顺序表迭代器,所以它拥有顺序表迭代器的所有功能
private class ListItr extends Itr implements ListIterator<E> {
        // 增加了指定位置的迭代器
        ListItr(int index) {
            cursor = index;
        }

        // 是否有前向节点(元素)
        public boolean hasPrevious() {
            return cursor != 0;
        }

        // 访问前向节点
        // 因为链表不支持随机访问,所以访问前向节点也是从首节点开始遍历到游标位置的前一个节点,时间复杂度为O(n),迭代器虽然提供了这样的操作,但是我们平时使用时需要注意,尽量减少对链表的这种操作
        public E previous() {
            checkForComodification();
            try {
                int i = cursor - 1;
                E previous = get(i);
                lastRet = cursor = i;
                return previous;
            } catch (IndexOutOfBoundsException e) {
                checkForComodification();
                throw new NoSuchElementException();
            }
        }

        public int nextIndex() {
            return cursor;
        }

        public int previousIndex() {
            return cursor-1;
        }
        
        // 修改游标指示元素值
        public void set(E e) {
            if (lastRet < 0)
                throw new IllegalStateException();
            checkForComodification();

            try {
                AbstractList.this.set(lastRet, e);
                expectedModCount = modCount;
            } catch (IndexOutOfBoundsException ex) {
                throw new ConcurrentModificationException();
            }
        }

        // 此处体现了链表的优势
        // 在游标位置插入一个元素,时间复杂度为O(1),如果是顺序表,插入一个元素的时间复杂度为O(n)
        public void add(E e) {
            checkForComodification();

            try {
                int i = cursor;
                AbstractList.this.add(i, e);
                lastRet = -1;
                cursor = i + 1;
                expectedModCount = modCount;
            } catch (IndexOutOfBoundsException ex) {
                throw new ConcurrentModificationException();
            }
        }
    }

结论

1、迭代器封装了对List的操作,使得访问更安全、准确,增删元素不是简单地通过List实例remove或者add,还包含了并发校验等;
2、两种for循环都不能准确删除元素,如下方例子所示。

public static void main(String[] args) {
    List<Integer> digits = new ArrayList<>();
    digits.add(0);
    digits.add(1);
    digits.add(2);
        
    for (int i = 0; i < digits.size(); i++) {
        System.out.println("Size of list : " + digits.size());
        digits.remove(i);
    }
        
    for (Integer integer : digits) {
        digits.remove(integer);
    }
}

输出

Size of list : 3
Size of list : 2
Exception in thread "main" java.util.ConcurrentModificationException
    at java.util.ArrayList$Itr.checkForComodification(ArrayList.java:901)
    at java.util.ArrayList$Itr.next(ArrayList.java:851)
    at collection.ListRemove.main(ListRemove.java:19)

相关文章:

  • asp.net core ef core mysql 新增数据并发异常处理
  • xshell连接Ubuntu系统
  • 多维分析的后台性能优化手段
  • 企业级自动化运维工具应用实战-ansible
  • 网络编程 与 面向对象
  • MapGIS6.7投影生成线-以物化探综合剖面图为例
  • C# 编码命名规则
  • HttpClient超时机制(安全问题处理:访问超大文件控制)
  • 关于如何在ElementUI中实现统计Table筛选结果数量
  • iOS网络基础 实战进阶篇
  • 使用ELK构建分布式日志分析系统
  • 后端的一些经验与心得
  • 超过父控件的部分不能响应事件怎么办
  • WKWebView的使用总结(oc与js交互使用心得)
  • JavaScript 中的错误隔离
  • 《Java编程思想》读书笔记-对象导论
  • 「前端」从UglifyJSPlugin强制开启css压缩探究webpack插件运行机制
  • AWS实战 - 利用IAM对S3做访问控制
  • css选择器
  • Java 9 被无情抛弃,Java 8 直接升级到 Java 10!!
  • JavaScript设计模式与开发实践系列之策略模式
  • JAVA多线程机制解析-volatilesynchronized
  • Mysql5.6主从复制
  • Nodejs和JavaWeb协助开发
  • Spring声明式事务管理之一:五大属性分析
  • vue2.0一起在懵逼的海洋里越陷越深(四)
  • vue-cli在webpack的配置文件探究
  • webgl (原生)基础入门指南【一】
  • 案例分享〡三拾众筹持续交付开发流程支撑创新业务
  • 成为一名优秀的Developer的书单
  • 第2章 网络文档
  • 基于 Babel 的 npm 包最小化设置
  • 基于webpack 的 vue 多页架构
  • 如何胜任知名企业的商业数据分析师?
  • 为物联网而生:高性能时间序列数据库HiTSDB商业化首发!
  • 要让cordova项目适配iphoneX + ios11.4,总共要几步?三步
  • 源码安装memcached和php memcache扩展
  • 远离DoS攻击 Windows Server 2016发布DNS政策
  • ​力扣解法汇总1802. 有界数组中指定下标处的最大值
  • # Python csv、xlsx、json、二进制(MP3) 文件读写基本使用
  • #HarmonyOS:软件安装window和mac预览Hello World
  • #if #elif #endif
  • $ is not function   和JQUERY 命名 冲突的解说 Jquer问题 (
  • (1)(1.9) MSP (version 4.2)
  • (delphi11最新学习资料) Object Pascal 学习笔记---第2章第五节(日期和时间)
  • (Matalb回归预测)PSO-BP粒子群算法优化BP神经网络的多维回归预测
  • (PWM呼吸灯)合泰开发板HT66F2390-----点灯大师
  • (附源码)ssm捐赠救助系统 毕业设计 060945
  • (附源码)计算机毕业设计SSM疫情居家隔离服务系统
  • (六)库存超卖案例实战——使用mysql分布式锁解决“超卖”问题
  • (十六)串口UART
  • (四)Controller接口控制器详解(三)
  • (学习日记)2024.03.12:UCOSIII第十四节:时基列表
  • (循环依赖问题)学习spring的第九天
  • (原創) 如何動態建立二維陣列(多維陣列)? (.NET) (C#)