Java基础-----集合类(四)

文章目录

  • [1. Iterator和ListIterator](#1. Iterator和ListIterator)
    • [1.1 简介](#1.1 简介)
    • [1.2 常用方法](#1.2 常用方法)
  • [2. remove方法](#2. remove方法)
    • [2.1 比较foreach方式和迭代器方式删除元素](#2.1 比较foreach方式和迭代器方式删除元素)
    • [2.2 找原因 -- 迭代器删除操作源码](#2.2 找原因 -- 迭代器删除操作源码)

1. Iterator和ListIterator

1.1 简介

1.Iterator 可以遍历List集合,也可以遍历Set集合; ListIterator只能遍历List集合

2.Iterator 只能单向遍历(向后遍历),ListIterator双向遍历(向前/向后遍历)

3.ListIterator继承Iterator接口,添加新的方法

1.2 常用方法

  • add(E e):将指定的元素插入到集合中,插入位置为迭代器当前位置之前

  • hasNext():正向遍历集合,判断后面是否还有元素,返回boolean类型值

  • next():返回集合中迭代器指向后面位置的元素

  • nextIndex():返回集合中迭代器后面位置元素的索引

  • hasPrevious():反向遍历集合,判断前面是否还有元素,返回boolean类型值

  • previous():返回集合中迭代器指向前面位置的元素

  • previousIndex():返回集合中迭代器前面位置元素的索引

  • set(E e):替换迭代器当前位置的元素

java 复制代码
		List<String> list=new ArrayList<>();
        list.add("aa");
        list.add("bb");
        list.add("cc");
        list.add("dd");
        for (Iterator<String> iterator= list.iterator();iterator.hasNext();){
            System.out.println(iterator.next());
        }
java 复制代码
		List<String> list=new ArrayList<>();
        list.add("aa");
        list.add("bb");
        list.add("cc");
        list.add("dd");
		//正向遍历
        ListIterator<String > iterator=list.listIterator();
        iterator.add("EEE");
        for (String s : list) {
            System.out.println(s);//EEE aa bb cc dd
        }
java 复制代码
		List<String> list=new ArrayList<>();
        list.add("aa");
        list.add("bb");
        list.add("cc");
        list.add("dd");

		ListIterator<String > iterator=list.listIterator();
        iterator.next();
        iterator.add("EEE");
        for (String s : list) {
            System.out.println(s);//aa EEE bb cc dd
        }
        System.out.println("-------------------");
        iterator=list.listIterator();
        for (;iterator.hasNext();){
            System.out.println(iterator.next());
        }
        System.out.println("-------------------");
        iterator=list.listIterator();
        for (;iterator.hasNext();){
            System.out.println(iterator.nextIndex()+"\t"+iterator.next());
        }

运行结果:

java 复制代码
//反向遍历
        ListIterator<String> iterator=list.listIterator(list.size());
        while (iterator.hasPrevious()) {
            System.out.println(iterator.previousIndex()+"\t"+iterator.previous());
        }

运行结果:

java 复制代码
		ListIterator<String> iterator=list.listIterator(list.size());
        while (iterator.hasPrevious()) {
            System.out.println(iterator.previousIndex()+"\t"+iterator.previous());
        }
        iterator.set("GGG");
        System.out.println(list);//[GGG, bb, cc, dd]
----------------------------------------------------------------------------
        ListIterator<String> iterator=list.listIterator(list.size());
        iterator.previous();
        iterator.set("GGG");
        while (iterator.hasPrevious()) {
            System.out.println(iterator.previousIndex()+"\t"+iterator.previous());
        }

        System.out.println(list);//[aa, bb, cc, GGG]

2. remove方法

2.1 比较foreach方式和迭代器方式删除元素

对集合元素进行循环处理增加或删除时,不能使用foreach处理方式,要使用迭代器方式。

  • 使用foreach方法进行删除,报错:java.util.ConcurrentModificationException
java 复制代码
		ArrayList<String> list=new ArrayList<>();
        list.add("aaa");
        list.add("bbb");
        list.add("ccc");
        list.add("ddd");

        for (String s : list) {
            if ("aaa".equals(s)){
                list.remove(s);
            }
        }
        System.out.println(list);
  • 使用迭代器方式删除元素:
java 复制代码
		Iterator<String> iterator=list.iterator();
        while (iterator.hasNext()){
            String x=iterator.next();
            if (x.equals("aaa")){
                iterator.remove();
            }
        }
        System.out.println(list);//[bbb,ccc,ddd]
  • 但是,在foreach对集合中倒数第二个元素进行删除时,不会报错,其他位置的元素都会报错。
java 复制代码
		for (String s : list) {
            if ("ccc".equals(s)){
                list.remove(s);
            }
        }
        System.out.println(list);//[aaa,bbb,ddd]

其中,foreach底层也是通过迭代器实现的。

其实,我们在使用迭代器操作时,有两个步骤:

java 复制代码
iterator.hasNext();//判断是否有下一个元素

item=iterator.next();//有下一个元素的话,取出

2.2 找原因 -- 迭代器删除操作源码

首先,我们进入到Iterator的remove()方法中,找到Itr类,在这个类中我们会看到ArrayList的remove()方法


在remove()方法中,看到里面有个checkForComodification()方法

java 复制代码
		final void checkForComodification() {
            if (modCount != expectedModCount)
                throw new ConcurrentModificationException();
        }

在checkForComodification()方法中,有两个变量:modCount变量和expectedModCount变量,当这两个变量不相同时,就会抛出异常处理(也就是我们在前面遇到的那个异常)

java 复制代码
modCount变量:记录集合对象从new出来到现在被修改的次数

expectedModCount变量:迭代器现在期望这个集合被修改的次数

在上面源码中看到:expectedModCount = modCount;只要保持这两个变量相等,删除操作就没有问题。
所以,迭代器不会报错,使用foreach会报错,原因是:
在迭代器中,对这两个变量进行了同步处理,而foreach没有进行同步处理,导致会出现checkForComodification异常出现。


通过查看源码,有一个hasNext()方法,方法中要比较cursor和size

java 复制代码
		public boolean hasNext() {
            return cursor != size;
        }

cursor是下一个元素的索引值;size是整个集合元素个数

如果两个元素不相等,即有下一个元素,就返回true。

当删除倒数第二个元素时,cursor通过计算之后,得到cursor=size,导致迭代器认为不存在下一个元素,迭代结束。

整体架构如下:

相关推荐
APP 肖提莫2 分钟前
MyBatis-Plus分页拦截器,源码的重构(重构total总数的计算逻辑)
java·前端·算法
kirito学长-Java3 分钟前
springboot/ssm太原学院商铺管理系统Java代码编写web在线购物商城
java·spring boot·后端
爱学习的白杨树4 分钟前
MyBatis的一级、二级缓存
java·开发语言·spring
OTWOL10 分钟前
两道数组有关的OJ练习题
c语言·开发语言·数据结构·c++·算法
问道飞鱼13 分钟前
【前端知识】强大的js动画组件anime.js
开发语言·前端·javascript·anime.js
拓端研究室14 分钟前
R基于贝叶斯加法回归树BART、MCMC的DLNM分布滞后非线性模型分析母婴PM2.5暴露与出生体重数据及GAM模型对比、关键窗口识别
android·开发语言·kotlin
Code成立15 分钟前
《Java核心技术I》Swing的网格包布局
java·开发语言·swing
Auc2419 分钟前
使用scrapy框架爬取微博热搜榜
开发语言·python
中草药z20 分钟前
【Spring】深入解析 Spring 原理:Bean 的多方面剖析(源码阅读)
java·数据库·spring boot·spring·bean·源码阅读
QQ同步助手26 分钟前
C++ 指针进阶:动态内存与复杂应用
开发语言·c++