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

设计模式-迭代器模式(Iterator)

设计模式-迭代器模式(Iterator)

    • 一、迭代器模式概述
      • 1.1 什么是迭代器模式
      • 1.2 简单实现迭代器模式
      • 1.3 使用迭代器模式的注意事项
    • 二、迭代器模式的用途
    • 三、迭代器模式实现方式
      • 3.1 使用Iterator接口实现迭代器模式
      • 3.2 使用Iterable接口和Iterator接口实现迭代器模式
      • 3.3 使用匿名内部类实现迭代器模式
      • 3.4 使用Lambda表达式实现迭代器模式

一、迭代器模式概述

1.1 什么是迭代器模式

迭代器模式是一种设计模式,它用于提供一种方法来访问一个容器对象中的各个元素,而又不暴露该对象的内部表示。

在迭代器模式中,我们定义一个抽象的迭代器类,它包含两个方法:一个是hasNext()方法,用于判断是否还有下一个元素;另一个是next()方法,用于获取下一个元素。然后,每个容器类都实现自己的迭代器类,以访问容器中的元素。

1.2 简单实现迭代器模式

首先,我们定义一个接口Iterator,它表示一个迭代器:

interface Iterator {boolean hasNext();Object next();
}

然后,我们可以实现一些具体的迭代器类,例如数组迭代器和列表迭代器:

class ArrayIterator implements Iterator {private final int[] array;private int index;public ArrayIterator(int[] array) {this.array = array;this.index = 0;}@Overridepublic boolean hasNext() {return index < array.length;}@Overridepublic Object next() {return array[index++];}
}class ListIterator implements Iterator {private final List<Object> list;private int index;public ListIterator(List<Object> list) {this.list = list;this.index = 0;}@Overridepublic boolean hasNext() {return index < list.size();}@Overridepublic Object next() {return list.get(index++);}
}

接下来,我们可以使用这些迭代器类来遍历容器中的元素:

int[] array = {1, 2, 3, 4, 5};
ArrayIterator arrayIterator = new ArrayIterator(array);
while (arrayIterator.hasNext()) {System.out.println(arrayIterator.next()); // 输出:1, 2, 3, 4, 5
}
List<Object> list = new ArrayList<>();
list.add("apple");
list.add("banana");
list.add("orange");
ListIterator listIterator = new ListIterator(list);
while (listIterator.hasNext()) {System.out.println(listIterator.next()); // 输出:apple, banana, orange
}

1.3 使用迭代器模式的注意事项

  • 1、容器对象需要实现Iterable接口,并实现iterator()方法,该方法返回一个迭代器对象。

  • 2、迭代器对象需要实现Iterator接口,并实现hasNext()和next()方法。

  • 3、在使用迭代器遍历容器中的元素时,需要先调用hasNext()方法判断是否还有下一个元素,如果有再调用next()方法获取下一个元素。

  • 4、在遍历过程中,如果修改了容器中的元素,可能会导致迭代器失效或抛出异常。因此,在遍历过程中应该避免对容器进行修改操作。

  • 5、如果需要在遍历过程中删除容器中的元素,可以使用迭代器的remove()方法。但是需要注意的是,在使用remove()方法后,迭代器将指向被删除元素的下一个元素,而不是当前元素。因此,在使用remove()方法后,需要再次调用next()方法才能获取正确的元素。

  • 6、如果需要在遍历过程中添加元素到容器中,可以使用迭代器的add()方法。但是需要注意的是,在使用add()方法后,迭代器将指向新添加的元素,而不是当前元素。因此,在使用add()方法后,需要再次调用next()方法才能获取正确的元素。

二、迭代器模式的用途

  • 1、支持以不同的方式遍历一个聚合对象:迭代器模式将聚合对象的遍历行为分离出来,提供了多种遍历方式。

  • 2、简化了聚合类:通过引入迭代器,可以将遍历逻辑从聚合类中分离出来,使得聚合类的职责更加单一。

  • 3、增加了代码的灵活性和可扩展性:由于引入了抽象层,增加新的聚合类和迭代器类都很方便,无需修改原有代码。

  • 4、让外部代码能够透明地访问集合内部的数据:迭代器负责集合对象的遍历,可以让外部的代码无需关心集合的内部表示,而直接访问其中的数据。

三、迭代器模式实现方式

3.1 使用Iterator接口实现迭代器模式

要使用Java中的Iterator接口实现迭代器模式,首先需要创建一个实现了Iterable接口的类要使用Java中的Iterator接口实现迭代器模式,首先需要创建一个实现了Iterable接口的类,然后在该类中实现iterator()方法。接下来,创建一个实现了Iterator接口的类,并在该类中实现hasNext()和next()方法。以下是一个简单的示例:

// 创建一个实现了Iterable接口的类
class MyIterable implements Iterable<String> {private String[] items;public MyIterable(String[] items) {this.items = items;}// 实现iterator()方法@Overridepublic Iterator<String> iterator() {return new MyIterator();}
}// 创建一个实现了Iterator接口的类
class MyIterator implements Iterator<String> {private int index;private String[] items;public MyIterator() {this.index = 0;}// 实现hasNext()方法@Overridepublic boolean hasNext() {return index < items.length;}// 实现next()方法@Overridepublic String next() {return items[index++];}
}public class Main {public static void main(String[] args) {String[] items = {"A", "B", "C", "D"};MyIterable myIterable = new MyIterable(items);for (String item : myIterable) {System.out.println(item);}}
}

在这个示例中,我们创建了一个名为MyIterable的类,它实现了Iterable接口。我们还创建了一个名为MyIterator的类,它实现了Iterator接口。在主方法中,我们创建了一个MyIterable对象,并使用for-each循环遍历其元素。

3.2 使用Iterable接口和Iterator接口实现迭代器模式

要使用Java中的Iterable接口和Iterator接口实现迭代器模式,首先需要创建一个类实现Iterable接口,然后在该类中实现iterator()方法。接下来,创建一个实现Iterator接口的类,并在该类中实现hasNext()和next()方法。以下是一个简单的示例:

// 创建一个实现Iterable接口的类
class MyIterable implements Iterable<Integer> {private int[] data;public MyIterable(int[] data) {this.data = data;}// 实现iterator()方法@Overridepublic Iterator<Integer> iterator() {return new MyIterator();}
}// 创建一个实现Iterator接口的类
class MyIterator implements Iterator<Integer> {private int[] data;private int index;public MyIterator() {this.index = 0;}// 实现hasNext()方法@Overridepublic boolean hasNext() {return index < data.length;}// 实现next()方法@Overridepublic Integer next() {return data[index++];}
}public class Main {public static void main(String[] args) {int[] data = {1, 2, 3, 4, 5};MyIterable myIterable = new MyIterable(data);for (int num : myIterable) {System.out.println(num);}}
}

在这个示例中,我们创建了一个名为MyIterable的类,它实现了Iterable接口。我们还创建了一个名为MyIterator的类,它实现了Iterator接口。在主方法中,我们创建了一个MyIterable对象,并使用for-each循环遍历其元素。

3.3 使用匿名内部类实现迭代器模式

在Java中,可以使用匿名内部类实现迭代器模式。以下是一个简单的示例:

import java.util.Iterator;public class AnonymousInnerClassIterator {public static void main(String[] args) {int[] numbers = {1, 2, 3, 4, 5};// 使用匿名内部类实现迭代器模式Iterator<Integer> iterator = new Iterator<Integer>() {private int index = 0;@Overridepublic boolean hasNext() {return index < numbers.length;}@Overridepublic Integer next() {return numbers[index++];}};// 遍历数组并打印元素while (iterator.hasNext()) {System.out.println(iterator.next());}}
}

在这个示例中,我们创建了一个匿名内部类实现了Iterator接口,并重写了hasNext()和next()方法。然后,我们使用这个匿名内部类的实例来遍历一个整数数组。

3.4 使用Lambda表达式实现迭代器模式

在Java中,可以使用Lambda表达式实现迭代器模式。以下是一个简单的示例:

首先,创建一个接口Iterator,包含两个方法:hasNext()和next()。

public interface Iterator<T> {boolean hasNext();T next();
}

然后,创建一个类MyIterator,实现Iterator接口。在这个类中,使用Lambda表达式定义hasNext()和next()方法。

import java.util.function.Supplier;public class MyIterator<T> implements Iterator<T> {private T[] items;private int index = 0;public MyIterator(T[] items) {this.items = items;}@Overridepublic boolean hasNext() {return index < items.length;}@Overridepublic T next() {return items[index++];}
}

最后,在主函数中使用MyIterator类。

public class Main {public static void main(String[] args) {Integer[] numbers = {1, 2, 3, 4, 5};MyIterator<Integer> iterator = new MyIterator<>(numbers);while (iterator.hasNext()) {System.out.println(iterator.next());}}
}

相关文章:

设计模式-迭代器模式(Iterator)

设计模式-迭代器模式&#xff08;Iterator&#xff09; 一、迭代器模式概述1.1 什么是迭代器模式1.2 简单实现迭代器模式1.3 使用迭代器模式的注意事项 二、迭代器模式的用途三、迭代器模式实现方式3.1 使用Iterator接口实现迭代器模式3.2 使用Iterable接口和Iterator接口实现迭…...

【计算机网络笔记】Internet网络的网络层——IP协议之IP数据报的结构

系列文章目录 什么是计算机网络&#xff1f; 什么是网络协议&#xff1f; 计算机网络的结构 数据交换之电路交换 数据交换之报文交换和分组交换 分组交换 vs 电路交换 计算机网络性能&#xff08;1&#xff09;——速率、带宽、延迟 计算机网络性能&#xff08;2&#xff09;…...

【Git】Git的GUI图形化工具ssh协议IDEA集成Git

一、GIT的GUI图形化工具 1、介绍 Git自带的GUI工具&#xff0c;主界面中各个按钮的意思基本与界面文字一致&#xff0c;与git的命令差别不大。在了解自己所做的操作情况下&#xff0c;各个功能点开看下就知道是怎么操作的。即使不了解&#xff0c;只要不做push操作&#xff0c;…...

Java中抽象类

1 抽象方法必须包含在抽象类中 package charactor; public abstract class Hero { String name; float hp;float armor;int moveSpeed;public static void main(String[] args) {}// 抽象方法attack // Hero的子类会被要求实现attack方法 public abstract void attack();} …...

18 Linux 阻塞和非阻塞 IO

一、阻塞和非阻塞 IO 1. 阻塞和非阻塞简介 这里的 IO 指 Input/Output&#xff08;输入/输出&#xff09;&#xff0c;是应用程序对驱动设备的输入/输出操作。当应用程序对设备驱动进行操作的时候&#xff0c;如果不能获取到设备资源&#xff0c;那么阻塞式 IO 就会将对应应用…...

多因素验证如何让企业邮箱系统登录更安全?

企业邮箱系统作为基础的办公软件之一&#xff0c;既是企业内外沟通的重要工具&#xff0c;也是连接企业多个办公平台的桥梁&#xff0c;往往涉及到客户隐私、业务信息、企业机密等等。为了保护邮箱账户的安全&#xff0c;设置登陆密码无疑是保护账户安全的常用措施之一。然而随…...

投票助手图文音视频礼物打赏流量主小程序开源版开发

投票助手图文音视频礼物打赏流量主小程序开源版开发 图文投票&#xff1a;用户可以发布图文投票&#xff0c;选择相应的选项进行投票。 音视频投票&#xff1a;用户可以发布音视频投票&#xff0c;观看音视频后选择相应的选项进行投票。 礼物打赏&#xff1a;用户可以在投票过…...

黑客(网络安全)技术——高效自学1.0

前言 前几天发布了一篇 网络安全&#xff08;黑客&#xff09;自学 没想到收到了许多人的私信想要学习网安黑客技术&#xff01;却不知道从哪里开始学起&#xff01;怎么学 今天给大家分享一下&#xff0c;很多人上来就说想学习黑客&#xff0c;但是连方向都没搞清楚就开始学习…...

8255 boot介绍及bring up经验分享

这篇文章会简单的介绍8255的启动流程&#xff0c;然后着重介绍8255在实际项目中新硬件上的bring up工作&#xff0c;可以给大家做些参考。 8255 boot介绍 下面这些信息来自文档&#xff1a;《QAM8255P IVI Boot and CoreBSP Architecture Technical Overview》 80-42847-11 R…...

visual studio 启用DPI识别功能

在开发widow程序时&#xff0c;有时必须将电脑 设置-->显示-->缩放与布局-->更改文本、应用项目的大小-->100%后&#xff0c;程序的画面才能正确运行&#xff0c;居说这是锁定了dpi的原因&#xff0c;需要启dpi识别功能。设置方法如下&#xff1a; 或者...

一题三解(暴力、二分查找算法、单指针):鸡蛋掉落

涉及知识点 暴力、二分查找算法、单指针 题目 给你 k 枚相同的鸡蛋&#xff0c;并可以使用一栋从第 1 层到第 n 层共有 n 层楼的建筑。 已知存在楼层 f &#xff0c;满足 0 < f < n &#xff0c;任何从 高于 f 的楼层落下的鸡蛋都会碎&#xff0c;从 f 楼层或比它低的…...

第一章 Object-XML 映射简介

文章目录 第一章 Object-XML 映射简介基础如何工作的映射选项IRIS 中的相关工具XML 文档的可能应用 第一章 Object-XML 映射简介 基础 将对象映射到 XML 一词意味着定义如何将该对象用作 XML 文档。要将对象映射到 XML&#xff0c;请将 %XML.Adaptor 添加到定义该对象的类的超…...

精密设备企业适合哪款CRM客户管理体系?

精密设备企业致力于打造现代化管理体系&#xff0c;以精密的仪器、精细的销售、精准的市场、精确的售后为企业核心&#xff0c;提供优质的精密产品和专业服务。随着企业的发展及市场发展需要&#xff0c;建立高效的客户关系管理体系势在必行。那么&#xff0c;精密设备企业适合…...

Rasa-笔记

1 Rasa环境搭建 笔者使用的Rasa版本是古早的1.10.7&#xff0c;python环境3.7。 1、安装miniconda 2、conda创建python3.7环境 3、安装TensorFlow和GPU相关 4、安装Rasa相关 2 Rasa笔记 3 Rasa报错 3.1 ValueError: Can’t patch loop of type <class ‘uvloop.Loop’&g…...

云架构师学习------腾讯云通识-存储与数据库

云架构师学习------腾讯云通识-存储与数据库 云架构师学习------腾讯云通识-存储与数据库存储基础存储服务对象存储-COS产品概述功能概览产品优势 云硬盘-CBS产品概述产品功能产品优势云硬盘类型 文件存储-CFS产品概述产品功能产品优势文件存储类型及性能规格存储类型性能与规格…...

蓝桥杯之模拟与枚举day1

Question1卡片(C/CA组第一题) 这个是一道简单的模拟枚举题目&#xff0c;只要把对应每次的i的各个位都提取出来&#xff0c;然后对应的卡片数目减去1即可。属于打卡题目。注意for循环的特殊使用即可 #include <iostream> using namespace std; bool solve(int a[],int n…...

深度学习 python opencv 动物识别与检测 计算机竞赛

文章目录 0 前言1 深度学习实现动物识别与检测2 卷积神经网络2.1卷积层2.2 池化层2.3 激活函数2.4 全连接层2.5 使用tensorflow中keras模块实现卷积神经网络 3 YOLOV53.1 网络架构图3.2 输入端3.3 基准网络3.4 Neck网络3.5 Head输出层 4 数据集准备4.1 数据标注简介4.2 数据保存…...

爱家房产网站源码 爱家房产网商业版 微信互动营销整合+手机触屏版+经纪人分销

房产网站源码手机访问自动转手机版修改修复如下&#xff1a; 1&#xff0c;修复手机版首页标题头部名称 2&#xff0c;修复手机版首页频道导航按钮 3&#xff0c;新增手机版广告位置显示方式 4&#xff0c;修复手机版首页内容显示样式 5&#xff0c;手机版头部背景颜色ic…...

招聘信息采集

首先&#xff0c;我们需要使用PHP的curl库来发送HTTP请求。以下是一个基本的示例&#xff1a; <?php // 初始化curl $ch curl_init();// 设置代理 curl_setopt($ch, CURLOPT_PROXY, "jshk.com.cn");// 设置URL curl_setopt($ch, CURLOPT_URL, "http://www…...

java开发宝典

Java命名规范 1&#xff1a;代码中的命名均不能以下划线或美元符号开始&#xff0c;也不能以下划线或美元符号结束。 反例&#xff1a;_name / __name / $name / name_ / name$ / name__ 。 2&#xff1a;禁止使用拼音和英文混合。 反例&#xff1a;DaZhePromotion [打折] / …...

wordpress后台更新后 前端没变化的解决方法

使用siteground主机的wordpress网站&#xff0c;会出现更新了网站内容和修改了php模板文件、js文件、css文件、图片文件后&#xff0c;网站没有变化的情况。 不熟悉siteground主机的新手&#xff0c;遇到这个问题&#xff0c;就很抓狂&#xff0c;明明是哪都没操作错误&#x…...

测试微信模版消息推送

进入“开发接口管理”--“公众平台测试账号”&#xff0c;无需申请公众账号、可在测试账号中体验并测试微信公众平台所有高级接口。 获取access_token: 自定义模版消息&#xff1a; 关注测试号&#xff1a;扫二维码关注测试号。 发送模版消息&#xff1a; import requests da…...

剑指offer20_链表中环的入口节点

链表中环的入口节点 给定一个链表&#xff0c;若其中包含环&#xff0c;则输出环的入口节点。 若其中不包含环&#xff0c;则输出null。 数据范围 节点 val 值取值范围 [ 1 , 1000 ] [1,1000] [1,1000]。 节点 val 值各不相同。 链表长度 [ 0 , 500 ] [0,500] [0,500]。 …...

【Web 进阶篇】优雅的接口设计:统一响应、全局异常处理与参数校验

系列回顾&#xff1a; 在上一篇中&#xff0c;我们成功地为应用集成了数据库&#xff0c;并使用 Spring Data JPA 实现了基本的 CRUD API。我们的应用现在能“记忆”数据了&#xff01;但是&#xff0c;如果你仔细审视那些 API&#xff0c;会发现它们还很“粗糙”&#xff1a;有…...

根据万维钢·精英日课6的内容,使用AI(2025)可以参考以下方法:

根据万维钢精英日课6的内容&#xff0c;使用AI&#xff08;2025&#xff09;可以参考以下方法&#xff1a; 四个洞见 模型已经比人聪明&#xff1a;以ChatGPT o3为代表的AI非常强大&#xff0c;能运用高级理论解释道理、引用最新学术论文&#xff0c;生成对顶尖科学家都有用的…...

Android Bitmap治理全解析:从加载优化到泄漏防控的全生命周期管理

引言 Bitmap&#xff08;位图&#xff09;是Android应用内存占用的“头号杀手”。一张1080P&#xff08;1920x1080&#xff09;的图片以ARGB_8888格式加载时&#xff0c;内存占用高达8MB&#xff08;192010804字节&#xff09;。据统计&#xff0c;超过60%的应用OOM崩溃与Bitm…...

分布式增量爬虫实现方案

之前我们在讨论的是分布式爬虫如何实现增量爬取。增量爬虫的目标是只爬取新产生或发生变化的页面&#xff0c;避免重复抓取&#xff0c;以节省资源和时间。 在分布式环境下&#xff0c;增量爬虫的实现需要考虑多个爬虫节点之间的协调和去重。 另一种思路&#xff1a;将增量判…...

均衡后的SNRSINR

本文主要摘自参考文献中的前两篇&#xff0c;相关文献中经常会出现MIMO检测后的SINR不过一直没有找到相关数学推到过程&#xff0c;其中文献[1]中给出了相关原理在此仅做记录。 1. 系统模型 复信道模型 n t n_t nt​ 根发送天线&#xff0c; n r n_r nr​ 根接收天线的 MIMO 系…...

论文笔记——相干体技术在裂缝预测中的应用研究

目录 相关地震知识补充地震数据的认识地震几何属性 相干体算法定义基本原理第一代相干体技术&#xff1a;基于互相关的相干体技术&#xff08;Correlation&#xff09;第二代相干体技术&#xff1a;基于相似的相干体技术&#xff08;Semblance&#xff09;基于多道相似的相干体…...

使用Spring AI和MCP协议构建图片搜索服务

目录 使用Spring AI和MCP协议构建图片搜索服务 引言 技术栈概览 项目架构设计 架构图 服务端开发 1. 创建Spring Boot项目 2. 实现图片搜索工具 3. 配置传输模式 Stdio模式&#xff08;本地调用&#xff09; SSE模式&#xff08;远程调用&#xff09; 4. 注册工具提…...