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

请添加图片描述

文章目录

  • 1. Iterator和ListIterator
    • 1.1 简介
    • 1.2 常用方法
  • 2. remove方法
    • 2.1 比较foreach方式和迭代器方式删除元素
    • 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):替换迭代器当前位置的元素

		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());}
		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}
		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());}

运行结果:
在这里插入图片描述

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

运行结果:
在这里插入图片描述

		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
		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);
  • 使用迭代器方式删除元素:
		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对集合中倒数第二个元素进行删除时,不会报错,其他位置的元素都会报错。
		for (String s : list) {if ("ccc".equals(s)){list.remove(s);}}System.out.println(list);//[aaa,bbb,ddd]

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

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

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

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

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

在这里插入图片描述
在这里插入图片描述

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

在这里插入图片描述

		final void checkForComodification() {if (modCount != expectedModCount)throw new ConcurrentModificationException();}

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

modCount变量:记录集合对象从new出来到现在被修改的次数expectedModCount变量:迭代器现在期望这个集合被修改的次数

在上面源码中看到:expectedModCount = modCount;只要保持这两个变量相等,删除操作就没有问题。

所以,迭代器不会报错,使用foreach会报错,原因是:

在迭代器中,对这两个变量进行了同步处理,而foreach没有进行同步处理,导致会出现checkForComodification异常出现。


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

		public boolean hasNext() {return cursor != size;}

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

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

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

整体架构如下:
在这里插入图片描述

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/593296.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

05-SpringCloud-RabbitMQ-概述

RabbitMQ 1.初识MQ 1.1.同步和异步通讯 微服务间通讯有同步和异步两种方式&#xff1a; 同步通讯&#xff1a;就像打电话&#xff0c;需要实时响应。 异步通讯&#xff1a;就像发邮件&#xff0c;不需要马上回复。 两种方式各有优劣&#xff0c;打电话可以立即得到响应&am…

Linux安装rabbitMq RPM安装 以及带延迟插件

rabbitmq安装 文档中rabbitmq下载链接 以及延迟插件 网盘下载 目前下载文件中版本已经过多个服务器安装测试 完全成功 1.安装执行 rpm -ivh openssl-libs-1.0.2k-19.el7.x86_64.rpm --force --nodeps rpm -ivh libnsl-2.34-28.el9_0.x86_64.rpm --force --nodeps rpm -ivh e…

京东商品详情API:数据分析和挖掘以优化销售策略

京东商品详情API提供的数据分析和挖掘功能可以帮助商家优化销售策略&#xff0c;提高销售额和用户转化率。以下是一些可能的应用场景&#xff1a; 商品关联分析&#xff1a;通过分析商品之间的关联规则&#xff0c;商家可以发现哪些商品经常一起被购买&#xff0c;从而制定捆绑…

c++ / day04

1. 整理思维导图 2. 全局变量&#xff0c;int monster 10000;定义英雄类hero&#xff0c;受保护的属性string name&#xff0c;int hp,int attck&#xff1b;公有的无参构造&#xff0c;有参构造&#xff0c;虚成员函数 void Atk(){blood-0;}&#xff0c;法师类继承自英雄类&a…

跟cherno手搓游戏引擎【1】:配置与入口点

环境配置&#xff1a; 编译环境&#xff1a;VS2019 创建两个项目&#xff1a; 设置Sandbox为启动项&#xff1a; 设置sandbox的配置属性-常规-输出目录\中间目录为如下&#xff1a; 预处理定义&#xff1a;为了配置一些只有windows才能用的函数。 设置YOTOEngin&#xff08;我…

【大数据HA】keepalived结合haproxy实现高可用的HMS

背景 上一篇实现了haproxy代理后端HMS服务实现高可用。但是对于haproxy还是单点故障&#xff0c;所以需要对haproxy进一步做HA&#xff0c;实现真正的后端服务的HA。 要实现haproxy的HA&#xff0c;需要使用到keepalived&#xff0c;使用keepalived是VIP虚拟IP服务&#xff0…

【2023 CCF 大数据与计算智能大赛】基于TPU平台实现超分辨率重建模型部署 基于预训练ESPCN的轻量化图像超分辨率模型TPU部署方案

2023 CCF 大数据与计算智能大赛 《基于TPU平台实现超分辨率重建模型部署》 作品名&#xff1a;基于预训练ESPCN的轻量化图像超分辨率模型TPU部署方案 队伍名&#xff1a;Absofastlutely 蒋松儒 计算机科学与技术系 硕士 南京大学 中国-江苏 kahsoltqq.com 吕欢欢 计算…

经典卷积神经网络-VGGNet

经典卷积神经网络-VGGNet 一、背景介绍 VGG是Oxford的Visual Geometry Group的组提出的。该网络是在ILSVRC 2014上的相关工作&#xff0c;主要工作是证明了增加网络的深度能够在一定程度上影响网络最终的性能。VGG有两种结构&#xff0c;分别是VGG16和VGG19&#xff0c;两者并…

01-SpringCloud微服务入门

1.认识微服务 随着互联网行业的发展&#xff0c;对服务的要求也越来越高&#xff0c;服务架构也从单体架构逐渐演变为现在流行的微服务架构。这些架构之间有怎样的差别呢&#xff1f; 1.1.单体架构 单体架构&#xff1a;将业务的所有功能集中在一个项目中开发&#xff0c;打…

如何利用Oracle官方网站不登录账号下载和安装非最新版本的JDK(版本自由选择)

一、JDK概述 JDK(Java Development Kit)是Java开发工具集,是针对Java编程语言的软件开发环境。它包含了Java编译器、JRE(Java运行时环境)以及其他一些用于开发、调试和测试Java应用程序的工具,是Java开发人员的必备工具。 二、JDK下载 进入Oracle官方网站,我们很容易发…

桌面天气预报软件 Weather Widget free mac特点介绍

Weather Widget free for Mac多种吸引人的小部件设计可供选择&#xff0c;可以随时了解天气&#xff01;还可以在Dock和菜单栏中为您提供简短的天气预报或当前状况的概述。 Weather Widget free for Mac软件介绍 始终在桌面上使用时尚的天气小部件来随时了解天气&#xff01;多…

逻辑回归(LR)----机器学习

基本原理 逻辑回归&#xff08;Logistic Regression&#xff0c;LR&#xff09;也称为"对数几率回归"&#xff0c;又称为"逻辑斯谛"回归。 logistic回归又称logistic 回归分析 &#xff0c;是一种广义的线性回归分析模型&#xff0c;常用于数据挖掘&#…

FA发放云桌面并与FC对接

&#xff08;7&#xff09;分配桌面&#xff08;该组为刚刚创建的域名用户和组&#xff09;&#xff0c;确认无误&#xff0c;直接发放 &#xff08;8&#xff09;可在任务中心查看发放的进度 3、FA的登录流程 &#xff08;1&#xff09;登录WI&#xff1a;客户端访问VLB&…

springcloud alibaba整合sentinel并结合dashboard控制面板设置规则

目录 一、springcloud alibaba整合sentinel二、采用代码方式设置流控规则三、结合dashboard控制面板设置规则3.1、准备工作3.2、设置全局异常处理3.3、编写测试接口3.4、结合dashboard控制面板设置规则3.4.1、流控规则设置并测试——QPS3.4.2、流控规则设置并测试——线程数3.4…

【Unity嵌入Android原生工程】

Unity嵌入Android原生工程 本章学习,Unity模块嵌入Android## 标题Unity导出Android工程创建Android Studio工程Unity嵌入到Andorid StudioAndroid原生代码跳转到Unity场景工作需要嵌入原生工程,并实现热更,记录一下 工具,Unity2023.3.14,Android Studio 2022.3.1 patch3 Un…

《JVM由浅入深学习【四】 2023-12-24》JVM由简入深学习提升分享

JVM由简入深学习提升分享四 1.JVM中java堆的特点及作用2. JVM中对象如何在堆内存中分配3. JVM堆内存中的对象布局 1.JVM中java堆的特点及作用 是线程共享的一块区域虚拟机启动时就创建了是虚拟机中内存占用很大的一块存放所有的实例对象和数组GC主要的作用区域可分为新生代&am…

初学者快速入门学习日语,PDF文档音频教学资料合集

一、资料描述 本套学习资料是很全面的&#xff0c;共有734份文件&#xff0c;包括PDF&#xff0c;PPT&#xff0c;表格&#xff0c;图片&#xff0c;音频等多种格式&#xff0c;可以作为初级日语的学习教材&#xff0c;也是非常适合初学者入门的&#xff0c;可以帮助大家快速的…

Nginx(十三) 配置文件详解 - 反向代理(超详细)

本篇文章主要讲ngx_http_proxy_module和ngx_stream_proxy_module模块下各指令的使用方法。 1. 代理请求 proxy_pass 1.1 proxy_pass 代理请求 Syntax: proxy_pass URL; Default: — Context: location, if in location, limit_except 设置代理服务器的协议和地址以…

使用Redis进行搜索

文章目录 构建反向索引 构建反向索引 在Begin-End区域编写 tokenize(content) 函数&#xff0c;实现文本标记化的功能&#xff0c;具体参数与要求如下&#xff1a; 方法参数 content 为待标记化的文本&#xff1b; 文本标记的实现&#xff1a;使用正则表达式提取全小写化后的…

初识Java并发,一问读懂Java并发知识文集(1)

&#x1f3c6;作者简介&#xff0c;普修罗双战士&#xff0c;一直追求不断学习和成长&#xff0c;在技术的道路上持续探索和实践。 &#x1f3c6;多年互联网行业从业经验&#xff0c;历任核心研发工程师&#xff0c;项目技术负责人。 &#x1f389;欢迎 &#x1f44d;点赞✍评论…