十五、行为型(迭代器模式)

迭代器模式(Iterator Pattern)

概念
迭代器模式是一种行为型设计模式,它用于提供一种顺序访问聚合对象(如集合、列表、数组)元素的方式,而无需暴露该对象的内部表示。迭代器模式将遍历行为从集合对象中分离出来,通过提供统一的接口来访问不同聚合对象中的元素。


应用场景

  1. 遍历集合对象:在系统中需要遍历不同类型的集合(如列表、栈、队列)时,迭代器模式可以提供一致的方式来遍历,而不必了解集合的内部结构。

  2. 封装集合的内部结构:迭代器隐藏了集合的具体实现细节,用户不需要关心集合的内部实现,例如数组的长度、链表的节点等,避免了直接操作集合对象。

  3. 解耦集合与算法:通过将遍历算法与集合本身分离,用户可以独立定义不同的迭代器,满足特定的遍历需求,如正序遍历、倒序遍历、跳步遍历等。


注意点

  • 并发修改问题:当集合在遍历过程中被修改(如新增或删除元素),可能会引发并发修改异常,需要特别注意在多线程场景下的使用。

  • 迭代器失效问题:一些容器在迭代时,如果容器内部结构改变(比如删除了一个元素),默认的迭代器会失效并抛出异常。

  • 内存管理:如果集合特别大且占用大量内存,使用迭代器遍历时应考虑如何管理内存,例如在遍历过程中是否需要保持所有元素在内存中。


核心要素

  1. Iterator(迭代器接口):定义访问和遍历元素的方法,例如next()hasNext()

  2. ConcreteIterator(具体迭代器):具体的迭代器实现,实现迭代器接口并维护遍历状态。

  3. Aggregate(聚合对象接口):定义创建迭代器的接口,用于返回具体的迭代器对象。

  4. ConcreteAggregate(具体聚合对象):具体的集合类,实现创建迭代器的接口,并负责存储对象。


Java代码完整示例

代码示例:简单的迭代器模式实现

import java.util.ArrayList;
import java.util.List;// 迭代器接口
interface Iterator<T> {boolean hasNext();T next();
}// 聚合对象接口
interface Aggregate<T> {Iterator<T> createIterator();
}// 具体迭代器实现
class ConcreteIterator<T> implements Iterator<T> {private List<T> items;private int position = 0;public ConcreteIterator(List<T> items) {this.items = items;}@Overridepublic boolean hasNext() {return position < items.size();}@Overridepublic T next() {return items.get(position++);}
}// 具体聚合对象实现
class ConcreteAggregate<T> implements Aggregate<T> {private List<T> items = new ArrayList<>();public void addItem(T item) {items.add(item);}@Overridepublic Iterator<T> createIterator() {return new ConcreteIterator<>(items);}
}// 客户端代码
public class IteratorPatternDemo {public static void main(String[] args) {ConcreteAggregate<String> aggregate = new ConcreteAggregate<>();aggregate.addItem("Item 1");aggregate.addItem("Item 2");aggregate.addItem("Item 3");Iterator<String> iterator = aggregate.createIterator();while (iterator.hasNext()) {System.out.println(iterator.next());}}
}

输出结果

Item 1
Item 2
Item 3

各种变形用法完整示例

  1. 倒序迭代器
    有时需要反向遍历集合,这时可以通过实现倒序迭代器来实现。

    代码示例:倒序迭代器

    class ReverseIterator<T> implements Iterator<T> {private List<T> items;private int position;public ReverseIterator(List<T> items) {this.items = items;this.position = items.size() - 1;}@Overridepublic boolean hasNext() {return position >= 0;}@Overridepublic T next() {return items.get(position--);}
    }public class ReverseIteratorPatternDemo {public static void main(String[] args) {ConcreteAggregate<String> aggregate = new ConcreteAggregate<>();aggregate.addItem("Item 1");aggregate.addItem("Item 2");aggregate.addItem("Item 3");Iterator<String> reverseIterator = new ReverseIterator<>(aggregate.createIterator().items);while (reverseIterator.hasNext()) {System.out.println(reverseIterator.next());}}
    }
    

    输出结果

    Item 3
    Item 2
    Item 1
    
  2. 迭代器失效和并发修改异常
    当集合在迭代过程中发生变化时,通常会抛出ConcurrentModificationException异常,尤其在使用Java自带的集合类时需要特别注意。

    代码示例:并发修改异常

    import java.util.ArrayList;
    import java.util.Iterator;
    import java.util.List;public class ConcurrentModificationDemo {public static void main(String[] args) {List<String> list = new ArrayList<>();list.add("Item 1");list.add("Item 2");list.add("Item 3");Iterator<String> iterator = list.iterator();while (iterator.hasNext()) {String item = iterator.next();System.out.println(item);// 在迭代过程中修改集合,抛出异常if ("Item 2".equals(item)) {list.remove(item);}}}
    }
    

    输出结果

    Item 1
    Item 2
    Exception in thread "main" java.util.ConcurrentModificationException
    
  3. 自定义迭代器与泛型支持
    迭代器模式可以通过泛型来实现对不同类型的集合进行遍历。

    代码示例:自定义泛型迭代器

    class GenericAggregate<T> implements Aggregate<T> {private List<T> items = new ArrayList<>();public void addItem(T item) {items.add(item);}@Overridepublic Iterator<T> createIterator() {return new ConcreteIterator<>(items);}
    }public class GenericIteratorPatternDemo {public static void main(String[] args) {GenericAggregate<Integer> intAggregate = new GenericAggregate<>();intAggregate.addItem(1);intAggregate.addItem(2);intAggregate.addItem(3);Iterator<Integer> intIterator = intAggregate.createIterator();while (intIterator.hasNext()) {System.out.println(intIterator.next());}GenericAggregate<String> stringAggregate = new GenericAggregate<>();stringAggregate.addItem("Item A");stringAggregate.addItem("Item B");Iterator<String> stringIterator = stringAggregate.createIterator();while (stringIterator.hasNext()) {System.out.println(stringIterator.next());}}
    }
    

    输出结果

    1
    2
    3
    Item A
    Item B
    
  4. 内部类实现迭代器
    Java标准库中的集合类,如ArrayList等,通常通过内部类实现自己的迭代器。

    代码示例:内部类实现迭代器

    class MyCollection<T> {private List<T> items = new ArrayList<>();public void addItem(T item) {items.add(item);}public Iterator<T> iterator() {return new MyIterator();}private class MyIterator implements Iterator<T> {private int index = 0;@Overridepublic boolean hasNext() {return index < items.size();}@Overridepublic T next() {return items.get(index++);}}
    }public class InnerClassIteratorDemo {public static void main(String[] args) {MyCollection<String> collection = new MyCollection<>();collection.addItem("Item X");collection.addItem("Item Y");Iterator<String> iterator = collection.iterator();while (iterator.hasNext()) {System.out.println(iterator.next());}}
    }
    

    输出结果

    Item X
    Item Y
    

总结
迭代器模式通过提供统一的遍历接口,隐藏了聚合对象的内部结构,使得代码能够适应多种不同的集合类型。在实际开发中,Java标准库已经内置了大量的迭代器实现(如ListSet等),我们可以直接使用或扩展这些实现,来满足不同的需求。

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

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

相关文章

单片机输出方波

从P1.0上输出一个方波,高电平5ms&#xff0c;低电平10ms. &#xff03;include〈reg51。h〉 unsigned char flag; sbit outP1^0&#xff1b; void main() &#xff5b; flag0&#xff1b; TMOD0X02; TH06&#xff1b; TL06; TR01&#xff1b; EA1&#xff1b; ET0…

Redis JSON介绍和命令大全

Redis JSON介绍和命令大全 Redis JSON先说说JSON是什么再说说JSON Path先推荐两个网站JSONPath JAVA clents Redis JSON 安装内存json命令语法命令url命令解释JSON.ARRAPPENDJSON.ARRINDEXJSON.ARRINSERTJSON.ARRLENJSON.ARRPOPJSON.ARRTRIMJSON.CLEARJSON.DEBUG MEMORYJSON.DE…

centOS部署Jenkins实现项目可持续自动化部署

个人看的是尚硅谷的视频&#xff0c;跟着实战&#xff0c;但因为视频是21年的&#xff0c;所以很容易出现jenkins插件不适配问题。 因而个人直接用较新版的jdk和jenkins. 先切换到root用户 sudo su一、安装jdk 先查询可安装版本 yum list java*安装jdk&#xff08;只复制圈…

【算法】归并排序概念及例题运用

&#x1f4e2;博客主页&#xff1a;https://blog.csdn.net/2301_779549673 &#x1f4e2;欢迎点赞 &#x1f44d; 收藏 ⭐留言 &#x1f4dd; 如有错误敬请指正&#xff01; &#x1f4e2;本文由 JohnKi 原创&#xff0c;首发于 CSDN&#x1f649; &#x1f4e2;未来很长&#…

爬虫日常实战

爬取美团新闻信息&#xff0c;此处采用两种方法实现&#xff1a; 注意点&#xff1a;因为此处的数据都是动态数据&#xff0c;所以一定要考虑好向下滑动数据包会更新的情况&#xff0c;不然就只能读取当前页即第一页数据&#xff0c;方法一通过更新ajax数据包网址页数&#xf…

vscode 预览markdown 文件

1. 点击左边扩展 2. 搜索“Markdown Preview Enhanced” 3. 选第一个安装即可 4. 重启vscode 5. 打开一个markdown 文件 6. 点击右上角的预览按钮

[mysql]mysql的全部单行函数

单行函数 几乎我们认识的语言都会对一些常用的功能进行,封装,有些叫函数,有些叫方法(Java),后期我们还可以自定义函数. 现在我们就当大家是没有语言基础,我们来从头开始讲.不过大家肯定接触过,中学说的函数,yf(x)f代表的就是function的缩写,这里其y2x1fx代表的就是封装的内容…

FileLink内外网文件交换——致力企业高效安全文件共享

随着数字化转型的推进&#xff0c;企业之间的文件交流需求日益增加。然而&#xff0c;传统的文件传输方式往往无法满足速度和安全性的双重要求。FileLink作为一款专注于跨网文件交换的工具&#xff0c;致力于为企业提供高效、安全的文件共享解决方案。 应用场景一&#xff1a;项…

C++大沥2019年真题——数字圈

Hi&#xff01;大家好&#xff01;Im#张亿&#xff0c;今天来讲C大沥2019年真题——数字圈 题目描述 当我们写数字时会发现有些数字有封闭区域&#xff0c;有的数字没有封闭区域。 数字 0 有一个封闭区域&#xff0c;数字 1、2、 3 都没有封闭区域&#xff0c;数字 4 有一个封…

word中的内容旋转90度

在vsto、Aspose.Words 中&#xff0c;默认没有直接的 API 可以让表格整体旋转 90 度。然而&#xff0c;我们可以通过一些方式来实现类似的效果&#xff0c;具体思路如下&#xff1a; 将表格插入到一个形状&#xff08;Shape&#xff09;或文本框中&#xff0c;然后旋转该形状。…

《RECONX: RECONSTRUCT ANY SCENE FROM SPARSEVIEWS WITH VIDEO DIFFUSION MODEL》论文阅读

论文地址&#xff1a;https://arxiv.org/pdf/2408.16767 项目地址&#xff1a;GitHub - liuff19/ReconX: ReconX: Reconstruct Any Scene from Sparse Views with Video Diffusion Model ---------------------------------------------------------------------------------…

2019年计算机网络408真题解析

第一题&#xff1a; 解析&#xff1a;OSI参考模型第5层完成的功能 首先&#xff0c;我们需要对OSI参考模型很熟悉&#xff1a;从下到上依次是&#xff1a;物理层-数据链路层-网络层- 运输层-会话层-表示层-应用层&#xff0c;由此可知&#xff0c;题目要问的是会话层的主要功能…

什么是感知与计算融合?

感知与计算融合&#xff08;Perception-Computing Fusion&#xff09;是指将感知技术&#xff08;如传感器、摄像头等&#xff09;与计算技术&#xff08;如数据处理、人工智能等&#xff09;有机结合&#xff0c;以实现对环境的更深层次理解和智能反应的过程。该技术广泛应用于…

基于SSM品牌银饰售卖系统的设计

管理员账户功能包括&#xff1a;系统首页&#xff0c;个人中心&#xff0c;用户管理&#xff0c;促销活动管理&#xff0c;饰品管理&#xff0c;我的收藏管理&#xff0c;系统管理&#xff0c;订单管理 用户账号功能包括&#xff1a;系统首页&#xff0c;个人中心&#xff0c;…

新书速览|Android智能座舱开发:从源码到实践

《Android智能座舱开发:从源码到实践》 本书内容 《Android智能座舱开发:从源码到实践》是一本专注于Android智能座舱系统开发与优化的实战指南。《Android智能座舱开发:从源码到实践》共9章&#xff0c;第1章从搭建源码编译环境开始&#xff0c;详细指导读者如何下载和编译An…

活体人脸识别技术总结及实践

文章目录 1、背景2、人脸反伪装技术2.1 活体人脸识别常见模式2.2 学术上反伪装研究 3、工程实现3.1 Silent-Face3.2 Silent-Face模型转rknn3.3 Silent-Face模型的限制 1、背景 1.1 什么是活体检测&#xff1f; 在人脸识别之前&#xff0c;先判断一下屏幕前摄像头捕捉到的人脸是…

深度解析RLS(Recursive Least Squares)算法

目录 一、引言二、RLS算法的基本思想三、RLS算法的数学推导四、RLS算法的特点五、RLS算法的应用场景六、RLS算法的局限性七、总结 一、引言 在自适应滤波领域&#xff0c;LMS&#xff08;Least Mean Squares&#xff09;算法因其计算简单、实现方便而广受欢迎。然而&#xff0…

【leetcode|哈希表、动态规划】最长连续序列、最大子数组和

目录 最长连续序列 解法一&#xff1a;暴力枚举 复杂度 解法二&#xff1a;优化解法一省去二层循环中不必要的遍历 复杂度 最大子数组和 解法一&#xff1a;暴力枚举 复杂度 解法二&#xff1a;贪心 复杂度 解法三&#xff1a;动态规划 复杂度 最长连续序列 输入输…

【数据结构与算法】时间、空间复杂度详解

大家有没有遇到过&#xff0c;为什么有些程序跑得飞快&#xff0c;而有些程序却慢得让人抓狂&#xff1f;我们可能都是这样认为的&#xff1a;他写的程序效率高等等&#xff0c;确实如此。但这背后隐藏着两个重要的概念&#xff1a;时间复杂度和空间复杂度。它们就像程序的“效…

算法题总结(十九)——图论

图论 DFS框架 void dfs(参数) { if (终止条件) {存放结果;return; }for (选择&#xff1a;本节点所连接的其他节点) {处理节点;dfs(图&#xff0c;选择的节点); // 递归回溯&#xff0c;撤销处理结果 } }深搜三部曲 确认递归函数&#xff0c;参数确认终止条件处理目前搜索节…