JAVA小知识22:迭代器iterator与列表迭代器ListIterator

2024-06-17 13:04

本文主要是介绍JAVA小知识22:迭代器iterator与列表迭代器ListIterator,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

一、迭代器

在 Java 中,迭代器是一个设计模式,用于遍历集合中的元素。Java 提供了 Iterator 接口来实现这一功能。迭代器的主要作用是提供一种通用的遍历集合的方法,而不需要了解集合的具体实现细节。

1.1、迭代器的特点

  • 统一接口:提供统一的遍历接口,可以对各种集合进行遍历而不需要了解其具体实现。
  • 无序性:迭代器返回元素的顺序由集合的具体实现决定。
  • 一次性使用:迭代器是一次性的,使用完毕后不能复用。
  • 动态修改支持:在迭代过程中,可以通过迭代器的 remove方法安全地移除元素,而不会引发并发修改异常。

1.2、迭代器的优缺点

优点
  • 统一遍历方式:提供了一种通用的方法来遍历各种类型的集合。
  • 隐藏实现细节:不需要了解集合的内部结构,只需使用迭代器的接口即可遍历集合。
  • 安全删除:可以在迭代过程中安全地删除元素,避免并发修改异常。
  • 增强的灵活性:可以在任何地方停止和恢复遍历。
缺点
  • 单向遍历:Iterator 只能单向遍历集合,不能反向遍历。
  • 一次性使用:迭代器只能使用一次,不能复用。如果需要再次遍历,需要重新获取新的迭代器。
  • 有限的修改操作:只能通过 remove方法删除元素,不能添加或修改元素。如果需要更复杂的修改操作,需要使用其他工具或方法。
  • 可能的并发问题:虽然迭代器本身是线程不安全的,在多线程环境下需要额外的同步措施来避免并发问题。

二、迭代器的使用

在这里插入图片描述

2.1、简单的使用:

List<String> list = new ArrayList<>();
list.add("a");
list.add("b");
list.add("c");
// 获取迭代器
Iterator<String> iterator = list.iterator();
// 判断当前位置是否有元素
while(iterator.hasNext()) {
// 获取当前元素的值并且将迭代器移入下一个对象String next = iterator.next();System.out.println(next);
}

2.2、NoSuchElementException

看如下代码:

List<String> list = new ArrayList<>();
list.add("a");
list.add("b");
list.add("c");
// 获取迭代器
Iterator<String> iterator = list.iterator();
// 判断当前位置是否有元素
while(iterator.hasNext()) {
// 获取当前元素的值并且将迭代器移入下一个对象String next = iterator.next();System.out.println(next);
}
// 再次获取一次元素iterator.next();

iterator.hasNext()判断当前位置没有元素,while循环以及结束后,此时如果再次调用iterator.next()获取当前元素,会报出NoSuchElementException:没有当前元素异常。

2.3、注意事项

  1. 迭代器迭代完毕之后,迭代器指针是不会复位的,会一直停留在最后一个元素的位置,如果需要再次遍历,就需要重新获取迭代器对象。
  2. 循环中只能使用一次next方法。如果使用了两次也会因为最后一次超出范围而报NoSuchElementException异常
  3. 迭代器遍历的时候不能使用集合的增加或者删除的方法。

2.4、remove:删除

注意事项中说到:迭代器遍历的时候不能使用集合的增加或者删除的方法,那迭代器不就成打印的了?那要他有啥用,别急,迭代器提供了一个删除的方法:remove( ),它能移除集合中由迭代器返回的最后一个元素。

List<String> list = new ArrayList<>();
list.add("a");
list.add("b");
list.add("c");
// 获取迭代器
Iterator<String> iterator = list.iterator();
// 判断当前位置是否有元素
while(iterator.hasNext()) {
// 获取当前元素的值并且将迭代器移入下一个对象String next = iterator.next();if(next.startsWith("a")) {iterator.remove();}
}
System.out.println(list);//[b, c]

2.5、增强for循环

以防有人不知道增强for循环,这里先给出一个例子:

List<String> list = new ArrayList<>();
list.add("a");
list.add("b");
list.add("c");
for (String s : list) {System.out.println(s);
}

为什么这里要讲增强for循环?因为增强for循环的底层就是迭代器iterator,同理,他有如下特点:

  • 有限的操作能力:无法在遍历过程中修改集合(如删除元素),否则会抛出 ConcurrentModificationException 异常。
  • 无法访问索引:增强 for 循环不提供访问当前元素索引的能力,如果需要索引,仍需使用传统的 for 循环。
  • 只读遍历:增强 for循环仅适用于遍历,不能在遍历过程中执行复杂的操作(如移除元素等)。

三、列表迭代器ListIterator

ListIterator 是 Java 提供的一个专门用于列表的数据结构(如 ArrayList、LinkedList)的迭代器接口,它是 Iterator 接口的子接口。与Iterator相比,ListIterator 提供了更丰富的遍历和操作列表的功能,包括双向遍历添加元素替换元素等。

3.1、成员方法

方法签名返回类型说明
boolean hasNext()boolean如果迭代器后面还有元素,返回 true
E next()E返回迭代器的下一个元素。
boolean hasPrevious()boolean如果迭代器前面还有元素,返回 true
E previous()E返回迭代器的前一个元素。
int nextIndex()int返回迭代器的下一个元素的索引。
int previousIndex()int返回迭代器的前一个元素的索引。
void remove()void移除迭代器返回的最后一个元素。
void set(E e)void用指定的元素替换迭代器返回的最后一个元素。
void add(E e)void在列表中插入指定的元素。

3.2、方法解释

对于上述方法,我们只需要了解到add(),set()即可

add()
List<String> list = new ArrayList<>();
list.add("a");
list.add("b");
list.add("c");
// 获取迭代器
ListIterator<String> iterator = list.listIterator();
// 判断当前位置是否有元素
while (iterator.hasNext()) {if("a".equals(iterator.next())){iterator.add("a1");}
}
System.out.println(list);//[a, a1, b, c]
set()
List<String> list = new ArrayList<>();
list.add("a");
list.add("b");
list.add("c");
// 获取迭代器
ListIterator<String> iterator = list.listIterator();
// 判断当前位置是否有元素
while (iterator.hasNext()) {if("a".equals(iterator.next())){iterator.set("a1");}
}
System.out.println(list);//[a1, b, c]

这篇关于JAVA小知识22:迭代器iterator与列表迭代器ListIterator的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1069519

相关文章

Java内存泄漏问题的排查、优化与最佳实践

《Java内存泄漏问题的排查、优化与最佳实践》在Java开发中,内存泄漏是一个常见且令人头疼的问题,内存泄漏指的是程序在运行过程中,已经不再使用的对象没有被及时释放,从而导致内存占用不断增加,最终... 目录引言1. 什么是内存泄漏?常见的内存泄漏情况2. 如何排查 Java 中的内存泄漏?2.1 使用 J

JAVA系统中Spring Boot应用程序的配置文件application.yml使用详解

《JAVA系统中SpringBoot应用程序的配置文件application.yml使用详解》:本文主要介绍JAVA系统中SpringBoot应用程序的配置文件application.yml的... 目录文件路径文件内容解释1. Server 配置2. Spring 配置3. Logging 配置4. Ma

Java 字符数组转字符串的常用方法

《Java字符数组转字符串的常用方法》文章总结了在Java中将字符数组转换为字符串的几种常用方法,包括使用String构造函数、String.valueOf()方法、StringBuilder以及A... 目录1. 使用String构造函数1.1 基本转换方法1.2 注意事项2. 使用String.valu

C#使用yield关键字实现提升迭代性能与效率

《C#使用yield关键字实现提升迭代性能与效率》yield关键字在C#中简化了数据迭代的方式,实现了按需生成数据,自动维护迭代状态,本文主要来聊聊如何使用yield关键字实现提升迭代性能与效率,感兴... 目录前言传统迭代和yield迭代方式对比yield延迟加载按需获取数据yield break显式示迭

java脚本使用不同版本jdk的说明介绍

《java脚本使用不同版本jdk的说明介绍》本文介绍了在Java中执行JavaScript脚本的几种方式,包括使用ScriptEngine、Nashorn和GraalVM,ScriptEngine适用... 目录Java脚本使用不同版本jdk的说明1.使用ScriptEngine执行javascript2.

Spring MVC如何设置响应

《SpringMVC如何设置响应》本文介绍了如何在Spring框架中设置响应,并通过不同的注解返回静态页面、HTML片段和JSON数据,此外,还讲解了如何设置响应的状态码和Header... 目录1. 返回静态页面1.1 Spring 默认扫描路径1.2 @RestController2. 返回 html2

Spring常见错误之Web嵌套对象校验失效解决办法

《Spring常见错误之Web嵌套对象校验失效解决办法》:本文主要介绍Spring常见错误之Web嵌套对象校验失效解决的相关资料,通过在Phone对象上添加@Valid注解,问题得以解决,需要的朋... 目录问题复现案例解析问题修正总结  问题复现当开发一个学籍管理系统时,我们会提供了一个 API 接口去

Java操作ElasticSearch的实例详解

《Java操作ElasticSearch的实例详解》Elasticsearch是一个分布式的搜索和分析引擎,广泛用于全文搜索、日志分析等场景,本文将介绍如何在Java应用中使用Elastics... 目录简介环境准备1. 安装 Elasticsearch2. 添加依赖连接 Elasticsearch1. 创

Spring核心思想之浅谈IoC容器与依赖倒置(DI)

《Spring核心思想之浅谈IoC容器与依赖倒置(DI)》文章介绍了Spring的IoC和DI机制,以及MyBatis的动态代理,通过注解和反射,Spring能够自动管理对象的创建和依赖注入,而MyB... 目录一、控制反转 IoC二、依赖倒置 DI1. 详细概念2. Spring 中 DI 的实现原理三、

SpringBoot 整合 Grizzly的过程

《SpringBoot整合Grizzly的过程》Grizzly是一个高性能的、异步的、非阻塞的HTTP服务器框架,它可以与SpringBoot一起提供比传统的Tomcat或Jet... 目录为什么选择 Grizzly?Spring Boot + Grizzly 整合的优势添加依赖自定义 Grizzly 作为