设计模式详解(十七)——迭代子模式

迭代子模式简介

迭代子模式定义
迭代子模式,也称为迭代器模式,是一种对象行为模式。它的定义是:提供一种方法顺序访问一个聚合对象中各个元素,而又不需暴露该对象的内部表示。在迭代子模式中,迭代子是一个对象,它知道如何遍历集合,并且跟踪它当前在集合中的位置。迭代子提供了一种统一的方式来访问集合中的元素,无论集合的具体类型如何。迭代子模式的核心思想是将遍历集合的行为从集合本身中分离出来,这样集合就可以专注于自己的数据结构,而遍历集合的逻辑可以被封装在迭代子中。
迭代子模式在实际开发中经常用于处理集合类的数据,例如数组、列表、树等。它提供了一种灵活的方式来遍历集合中的元素,同时也使得代码更易于维护和扩展。通过将遍历集合的逻辑抽象出来,迭代子模式使得集合和遍历逻辑可以相互独立地变化,从而提高了代码的灵活性和可维护性。

迭代子模式包含以下角色:

  1. 抽象迭代子(Iterator):这是一个抽象角色,它定义了遍历元素所需的接口。迭代器的主要职责是提供一种方法来访问聚合对象中的元素,同时隐藏了聚合对象的内部表示。迭代器通常包含诸如获取下一个元素、判断是否还有更多元素等操作。
  2. 具体迭代子(ConcreteIterator):这个角色实现了迭代器接口,负责具体的遍历逻辑。它通常维护一个指向当前元素的游标,并在遍历过程中更新该游标的位置。具体迭代器需要了解聚合对象的内部结构,以便正确遍历其元素。
  3. 聚合对象(Aggregate):这也是一个抽象角色,它定义了创建迭代器对象的接口。聚合对象通常包含一组元素,并提供一个方法来获取一个迭代器对象,以便外部可以遍历这些元素。聚合对象不需要暴露其内部表示,只需提供获取迭代器的接口即可。
  4. 具体聚合对象(ConcreteAggregate):这个角色实现了聚合接口,负责创建具体的迭代器对象。具体聚合通常包含一组具体的元素,并实现了获取迭代器的方法,该方法返回一个与具体聚合对象相匹配的迭代器实例。

迭代子模式优缺点:
优点:

  1. 多态迭代:它为不同的聚合结构提供了一致的遍历接口,使得一个迭代接口可以访问不同的聚集对象。
  2. 简化聚集对象接口:通过将遍历逻辑封装在迭代器中,聚合对象的接口可以更加简洁和专一。聚合对象只需要提供一个创建迭代器的方法,而无需暴露其内部结构和遍历逻辑。
  3. 元素迭代功能多样化:每个聚集对象都可以提供一个或多个不同的迭代器,这使得同种元素聚合结构可以有不同的迭代行为。
  4. 提高代码的可读性和可维护性:使用迭代子模式可以使代码更加清晰和易于理解。遍历逻辑被封装在迭代器中,使得聚合对象的代码更加专注于其数据结构和核心功能。同时,由于迭代器的独立性,修改遍历逻辑时不会影响到聚合对象的代码,提高了代码的可维护性。
  5. 支持多种遍历方式:迭代子模式允许在不修改聚合对象的前提下增加新的遍历方式。通过实现不同的迭代器,可以方便地支持正序、逆序、过滤等多种遍历需求。

缺点:

  1. 实现复杂性:对于简单的遍历,如数组或有序列表,使用迭代子模式可能会增加实现的复杂性。所以对于小型项目或简单的数据结构,使用迭代子模式可能过于复杂和繁琐。
  2. 类的数量增加:每次新增一个集合类,都需要增加相应的迭代器类,这可能会导致类的个数成对增加,增加了系统的复杂性,增加代码的复杂性和管理成本。
  3. 性能开销:在某些情况下,使用迭代子模式可能会引入一些性能开销。由于每次遍历都需要创建新的迭代器对象,这可能会增加内存消耗和对象创建/销毁的开销。然而,这种开销通常是可以接受的,尤其是在需要灵活遍历聚合对象的情况下。

使用场景

  1. 访问聚合对象内容而不暴露内部表示:当需要访问一个聚合对象的内容,但不希望直接暴露其内部数据结构时,可以使用迭代子模式。通过迭代器,可以在不了解聚合对象内部实现细节的情况下,遍历并访问其元素。
  2. 为聚合对象提供多种遍历方式:当需要为聚合对象提供不同的遍历方式时,如正向遍历、逆向遍历、跳跃遍历等,迭代子模式非常适用。通过实现不同的迭代器,可以轻松支持多种遍历需求。
  3. 统一遍历不同聚合结构:当需要为不同类型的聚合结构提供一个统一的遍历接口时,迭代子模式可以发挥作用。无论是列表、栈、树还是图等数据结构,都可以通过迭代器进行统一的遍历操作,从而简化代码和提高可维护性。
  4. 隐藏集合对象的内部结构:当需要隐藏集合对象的内部结构,只提供统一的遍历接口时,迭代子模式非常有用。这样,外部代码可以透明地访问集合内部的数据,而无需了解集合的具体实现细节。

以下举一个迭代子模式的例子:
下面通过一个简单的例子来演示。

创建抽象聚合对象(Aggregate)

/**
 * 抽象聚合对象:定义了创建迭代器对象的接口
 */
public interface Aggregate {

    public void add(String str);

    public void remove(String str);

    public Iterator getIterator();
}

创建具体聚合对象(ConcreteAggregate)

import java.util.ArrayList;
import java.util.List;

/**
 * 具体聚合对象:实现了聚合接口,负责创建具体的迭代器对象
 */
public class ConcreteAggregate implements Aggregate {
    private List<String> list=new ArrayList<String>();
    public void add(String str)
    {
        list.add(str);
    }
    public void remove(String str)
    {
        list.remove(str);
    }
    public Iterator getIterator()
    {
        return(new ConcreteIterator(list));
    }

}

创建抽象迭代子(Iterator)

/**
 * 抽象迭代子:抽象角色,它定义了遍历元素所需的接口
 */
public interface Iterator {
    String first();
    String next();
    boolean hasNext();
}

创建具体迭代子(ConcreteIterator)

import java.util.List;

/**
 * 具体迭代子:实现了迭代器接口,负责具体的遍历逻辑
 */
public class ConcreteIterator implements Iterator {
    private List<String> list;
    private int index;

    public ConcreteIterator(List<String> list)
    {
        this.list=list;
    }

    // 判断列表对象中是否还有参数
    public boolean hasNext()
    {
        if(index < list.size()-1)
        {
            return true;
        }
        else
        {
            return false;
        }
    }

    // 获取对象列表中第一个参数
    public String first()
    {
        index = 0;
        String str = list.get(index);;
        return str;
    }

    // 获取对象列表中下一个参数
    public String next()
    {
        String str = null;
        if(this.hasNext())
        {
            str=list.get(++index);
        }
        return str;
    }
}

创建客户端(Client)

public class Client {
    public static void main(String[] args) {
        Aggregate aggregate = new ConcreteAggregate();
        aggregate.add("篮球");
        aggregate.add("足球");
        aggregate.add("羽毛球");
        aggregate.add("网球");
        System.out.println("添加后的聚合对象内容有:");

        Iterator iterator = aggregate.getIterator();
        while (iterator.hasNext()) {
            String str = iterator.next();
            System.out.println(str);
        }

        System.out.println("===================");
        String str = iterator.first();
        System.out.println("聚合对象里第一个内容是:" + str);
    }
}

输出结果如下所示:

添加后的聚合对象内容有:
足球
羽毛球
网球
===================
聚合对象里第一个内容是:篮球

在上述例子中,我们定义了一个迭代子模式,包括抽象聚合对象、具体聚合对象、抽象迭代子和具体迭代子。Aggregate类中定义了添加元素、移除元素和获取迭代器对象的接口。在这个例子中,它包含一个添加字符串的add方法,一个移除字符串的remove方法,以及一个返回迭代器对象的getIterator方法。ConcreteAggregate 类实现了 Aggregate 接口,并使用一个 ArrayList 来存储元素。Iterator类中定义了迭代器的行为,定义了遍历元素所需的接口,包括first(获取第一个元素)、next(获取下一个元素)和hasNext(判断是否还有更多元素)方法。ConcreteIterator 类实现了 Iterator 接口,并提供了具体的遍历逻辑。客户端代码 (Client 类) 则是创建了一个 ConcreteAggregate对象,添加了一些元素,并通过迭代器遍历这些元素。

总而言之:
迭代子模式是一种行为设计模式,通过将遍历集合对象的行为封装在迭代子对象中,实现了遍历与集合对象的分离。它包括抽象迭代子、具体迭代子、聚合对象和具体聚合对象等角色。优点包括提高代码灵活性、模块化和隐藏集合对象内部结构;缺点是增加了代码复杂度和可能降低性能。适用于需要统一遍历接口、隐藏集合对象细节、提供多种遍历方式等场景。迭代子模式能够使代码更易维护、灵活,并提供一种统一的遍历方式,是一种常用的设计模式。
总之,迭代子模式通过将遍历集合的行为从集合对象中分离出来,实现了更好的封装和更灵活的设计。


以上代码下载请点击该链接:https://github.com/Yarrow052/Java-package.git

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

【ai相关】人工智能的概念

一、人工智能的定义 人工智能&#xff0c;简称AI&#xff0c;是指由机器或计算机系统所展现出的类似于人类智能的行为和能力。其核心在于使机器能够像人类一样进行思考和行动&#xff0c;而这些思考和行动都是基于理性的决策和判断。 什么是机器学习&#xff1f; 机器学习的核…

【蓝桥杯省赛真题40】python摘苹果 中小学青少年组蓝桥杯比赛 算法思维python编程省赛真题解析

目录 python摘苹果 一、题目要求 1、编程实现 2、输入输出 二、算法分析 三、程序编写 四、程序说明 五、运行结果 六、考点分析 七、 推荐资料 1、蓝桥杯比赛 2、考级资料 3、其它资料 python摘苹果 第十三届蓝桥杯青少年组python编程省赛真题 一、题目要求 &…

二维码如何分享照片?3步在线生成相册二维码

拍摄的照片怎样快速分享给其他人呢&#xff1f;传统的图片传输方式多通过微信、QQ、空间、微博等方式来实现分享&#xff0c;当需要分享给指定人员时或者需要分享的图片数量较多时&#xff0c;这些方式传递起来并不是特别的方便。想要实现大量图片的分享&#xff0c;选择生成相…

docker教程(详细)

0 背景 软件开发最大的麻烦事之一&#xff0c;就是环境配置。环境配置如此麻烦&#xff0c;换一台机器&#xff0c;就要重来一次&#xff0c;旷日费时。很多人想到&#xff0c;能不能从根本上解决问题&#xff0c;软件可以带环境安装&#xff1f;也就是说&#xff0c;安装的时…

【JAVA进阶篇教学】第五篇:Java多线程编程

博主打算从0-1讲解下java进阶篇教学&#xff0c;今天教学第五篇&#xff1a;Java多线程编程。 在Java编程中&#xff0c;使用多线程可以提高程序的并发性能&#xff0c;但是直接创建和管理线程可能会导致资源浪费和性能下降。Java提供了线程池来管理线程的生命周期和执行任务…

Activiti——将绘制的流程图存入act数据库并进行流程推进与状态流转

文章目录 前言流程图入库操作 RepositoryService项目结构数据库连接配置文件入库Java测试代码zip 方式进行流程的批量部署 流程启动 RuntimeService待处理任务查看 TaskService流程状态的扭转查询流程定义信息 RepositoryService查询正在执行的流程实例 RuntimeService已部署流…

Autosar AP的基本构成

1. 引言 Autosar AP的体系结构是怎样的呢&#xff1f;从整体的宏观的方向上划分&#xff0c;分为 1&#xff09;应用层&#xff0c;其中放置各种应用组件SWCs。2&#xff09;运行时基本功能软件族群&#xff0c;提供基本AutoSar基本软件中间件&#xff0c;比如经典的通信服务…

玩转nginx的配置文件3

1. limit_req_zone配置限流 limit_req_zone $binary_remote_addr zonemylimit:10m rate10r/s;upstream myweb {server 10.0.105.196:80 weight1 max_fails1 fail_timeout1;}server {listen 80;server_name localhost;location /login {limit_req zonemylimit;proxy_pass http:…

嵌入式UBoot如何跳转Kernel—uboot与linux交界分析

不知道你是否有这种感觉,就是学习了Uboot,学习了kernel,学习了安卓。但是有时候总感觉是各自孤立的,将三者连续不起来? • 不知道你是否在做启动方案的时候,在宏观上知道了整个启动链路流程,但是却在汪洋的代码中迷了路? 那么这篇文章必定对你有点用处。 如果没有,那请…

恭喜!喜提美国匹兹堡大学儿童医院访问学者邀请函

➡️【院校简介】 匹兹堡UPMC儿童医院该院是匹兹堡大学医学中心的一部分&#xff0c;也是大匹兹堡唯一一家专门护理26岁以下婴儿&#xff0c;儿童&#xff0c;青少年和年轻人的医院。该医院隶属于匹兹堡大学医学院&#xff0c;设有一个获得州级认证的一级儿科创伤中心&#xf…

Linux的自动化脚本:使用crul命令的从某个网站获取数据(从url获取数据),并将其写入一个文件中

目录 一、要求 二、思路 三、shell脚本实现演练 &#xff08;一&#xff09;脚本实现 &#xff08;二&#xff09;脚本代码说明 &#xff08;三&#xff09;脚本执行 &#xff08;四&#xff09;数据内容 一、要求 Linux的一个进程需要获取一个网站上的最新数据&#xf…

JavaScript:将input标签中的内容打印到控制台

使用浏览器进行开发时&#xff0c;按F12可以查看网页信息。 目标&#xff1a;实现将input标签中的内容&#xff0c;打印到控制台&#xff08;console&#xff09; HTML页面的关键代码实现&#xff1a; 登录功能&#xff1a; HTML代码&#xff1a; <div class"form-…

FastAPI从入门到实战(16)——请求参数汇总

FastAPI有各种各样的参数,包括: url参数(定义在url中的参数)param参数(使用url后面?xxxx)定义的参数body参数(在请求主体中携带的json参数)form参数(在请求主体中携带的web表单参数)cookie参数(在请求的cookie中携带的参数)file参数(客户端上传的文件) 1. url参数 from fas…

【A-034】基于SSH的电影订票系统(含论文)

【A-034】基于SSH的电影订票系统&#xff08;含论文&#xff09; 开发环境&#xff1a; Jdk7(8)Tomcat7(8)MySQLIntelliJ IDEA(Eclipse) 数据库&#xff1a; MySQL 技术&#xff1a; SpringStruts2HiberanteJSPJquery 适用于&#xff1a; 课程设计&#xff0c;毕业设计&…

户外指南——时代产物

分类 一级分类&#xff1a; 衣&#xff1a;除了上述提到的&#xff0c;还包括衣物的材质、款式多样性、与身份地位的关联等。 食&#xff1a;还包括饮食的文化内涵、地域特色、对特殊饮食需求的满足等。 住&#xff1a;还包括居住空间的合理布局、智能家居的应用、与自然环境…

mysql的约束和表关系

根据查询的结果&#xff0c;复制出一个新表 create table newTable AS select * from oldTable; create table newPeople AS select * from day2_test.people; 约束 引入&#xff1a;如果某一列如id列&#xff0c;有重复的数据&#xff0c;无法准确定位&#xff0c;有的列有空…

安装VCenter 7 对硬件资源的需求

安装VMware vCenter Server 7.x 对硬件资源的需求主要包括以下方面&#xff1a; 服务器硬件&#xff1a; 处理器&#xff1a;64位 x86架构&#xff0c;推荐采用多核CPU以支持高并发管理和运行多个虚拟机。具体数量取决于vCenter Server将管理的虚拟机规模及复杂度。内存&#x…

vue系统指令二

vue系统指令二 v-model:双向数据绑定 重点&#xff1a;双向数据绑定&#xff0c;只能用于表单元素&#xff0c;或者用于自定义组件。 之前的文章里&#xff0c;我们通过v-bind&#xff0c;给<input>标签绑定了data对象里的name属性。当data里的name的值发生改变时&#…

未来五十年,智能科技将如何改变传统行业格局?

未来五十年内&#xff0c;随着人工智能&#xff08;AI&#xff09;和智能科技的不断发展&#xff0c;许多行业将面临被取代的风险。虽然这种趋势可能会带来一些担忧&#xff0c;但也将为人类社会带来巨大的变革。下面将详细探讨哪些行业可能会在未来被智能科技所取代。 ▶ 制造…

攻防世界fileclude题解

攻防世界fileclude题解 ​​ 题目要求file1和file2参数不能为空 且file2这个文件内容值为hello ctf&#xff0c;用php://input 然后POST体内输入hello ctf即可满足这个if条件 满足这个条件后就会包含file1变量所指定的那个文件。用php伪协议来跨目录包含一下flag.php文件就可以…