程序员必知!迭代器模式的实战应用与案例分析

news2025/1/11 9:00:08

程序员必知!迭代器模式的实战应用与案例分析 - 程序员古德

迭代器模式提供了遍历聚合对象中各元素的方法,通过它无需了解其底层结构,例如,在电商平台商品列表中统计销售额时,可通过迭代器简化遍历过程,加总每个商品销售额,此模式使代码更简洁、易维护,且适用于不同类型的聚合对象,实现轻松扩展。

定义

程序员必知!迭代器模式的实战应用与案例分析 - 程序员古德

迭代器模式提供了一种遍历一个聚合对象中各个元素的方法,而不需要知道该对象的底层表示方式,它的主要目的是提供一个统一的、简洁的方式来遍历一个聚合对象的所有元素,而不需要关心该聚合对象的内部表示方式。

举一个业务中形象的例子来说明迭代器模式:假设有一个电商平台的商品列表,这个商品列表由多个商品组成,现在想要统计所有商品的销售额,可以使用迭代器模式来遍历所有商品并计算总销售额,具体来说,可以定义一个商品列表的聚合对象,然后定义一个迭代器,通过迭代器来遍历所有的商品对象,并对每个商品对象执行加总操作,最终得到所有商品的销售额,使用迭代器模式可以使得代码更加简洁、易于维护,并且可以轻松地扩展到其他类型的聚合对象。

代码案例

程序员必知!迭代器模式的实战应用与案例分析 - 程序员古德

反例

下面是一个未使用迭代器模式的反例代码,在未使用迭代器模式的情况下,要遍历一个聚合对象(比如一个列表或集合)并对其进行操作,通常可能会直接在client代码中处理聚合对象的内部表示,这样做会导致client代码与聚合对象的内部结构紧密耦合,违反了设计模式的封装原则,如下代码:

import java.util.ArrayList;  
import java.util.List;  
  
// 产品类  
class Product {  
    private String name;  
    private double price;  
  
    public Product(String name, double price) {  
        this.name = name;  
        this.price = price;  
    }  
  
    public String getName() {  
        return name;  
    }  
  
    public double getPrice() {  
        return price;  
    }  
  
    @Override  
    public String toString() {  
        return "Product{" +  
                "name='" + name + '\'' +  
                ", price=" + price +  
                '}';  
    }  
}  
  
// 产品目录类,未使用迭代器模式  
class ProductCatalog {  
    // 内部使用ArrayList存储产品  
    private List<Product> products = new ArrayList<>();  
  
    // 直接暴露内部列表,违反封装原则  
    public List<Product> getProducts() {  
        return products;  
    }  
  
    public void addProduct(Product product) {  
        products.add(product);  
    }  
}  
  
// 客户端代码  
public class Client {  
    public static void main(String[] args) {  
        ProductCatalog catalog = new ProductCatalog();  
        catalog.addProduct(new Product("Laptop", 999.99));  
        catalog.addProduct(new Product("Smartphone", 499.99));  
        catalog.addProduct(new Product("Tablet", 299.99));  
  
        // 客户端直接操作ProductCatalog的内部结构  
        for (Product product : catalog.getProducts()) {  
            System.out.println(product);  
        }  
  
        // 假设我们需要修改ProductCatalog的内部实现,比如使用LinkedList替代ArrayList  
        // 那么客户端代码也需要相应地进行修改,这违反了开闭原则  
    }  
}	

运行上面的代码会得到以下输出:

Product{name='Laptop', price=999.99}  
Product{name='Smartphone', price=499.99}  
Product{name='Tablet', price=299.99}

在这个例子中,ProductCatalog类直接暴露了它的内部列表products,client代码通过getProducts方法直接访问这个列表并进行遍历,这样做的问题是,如果ProductCatalog的内部实现发生变化(比如从ArrayList换成LinkedList),client代码也需要相应地进行修改,因为client代码已经与ProductCatalog的内部实现紧密耦合在一起了。

为了避免这种紧密耦合,应该使用迭代器模式来提供一个统一的遍历接口,从而隐藏聚合对象的内部表示,这样,即使聚合对象的内部实现发生变化,client代码也不需要修改,因为它只依赖于迭代器接口。

正例

下面是一个使用迭代器模式的正例代码,创建一个迭代器接口来定义遍历元素所需的方法,然后为聚合对象创建一个类来实现这个接口,这样做的好处是client代码可以只依赖于迭代器接口,而不是聚合对象的具体实现,从而实现了松耦合,如下代码:

import java.util.ArrayList;  
import java.util.Iterator;  
import java.util.List;  
  
// 产品类  
class Product {  
    private String name;  
    private double price;  
  
    public Product(String name, double price) {  
        this.name = name;  
        this.price = price;  
    }  
  
    public String getName() {  
        return name;  
    }  
  
    public double getPrice() {  
        return price;  
    }  
  
    @Override  
    public String toString() {  
        return "Product{" +  
                "name='" + name + '\'' +  
                ", price=" + price +  
                '}';  
    }  
}  
  
// 迭代器接口  
interface Iterator<T> {  
    boolean hasNext();  
    T next();  
}  
  
// 产品目录类,使用了迭代器模式  
class ProductCatalog implements Iterable<Product> {  
    // 内部使用ArrayList存储产品  
    private List<Product> products = new ArrayList<>();  
  
    public void addProduct(Product product) {  
        products.add(product);  
    }  
  
    // 创建并返回产品迭代器  
    @Override  
    public Iterator<Product> iterator() {  
        return new ProductIterator(products);  
    }  
  
    // 私有内部类实现迭代器接口  
    private class ProductIterator implements Iterator<Product> {  
        private List<Product> productList;  
        private int position;  
  
        public ProductIterator(List<Product> productList) {  
            this.productList = productList;  
        }  
  
        @Override  
        public boolean hasNext() {  
            return position < productList.size();  
        }  
  
        @Override  
        public Product next() {  
            if (hasNext()) {  
                return productList.get(position++);  
            } else {  
                return null; // 或者抛出异常  
            }  
        }  
    }  
}  
  
// 客户端代码  
public class Client {  
    public static void main(String[] args) {  
        ProductCatalog catalog = new ProductCatalog();  
        catalog.addProduct(new Product("Laptop", 999.99));  
        catalog.addProduct(new Product("Smartphone", 499.99));  
        catalog.addProduct(new Product("Tablet", 299.99));  
  
        // 使用迭代器遍历产品目录  
        for (Product product : catalog) {  
            System.out.println(product);  
        }  
  
        // 客户端代码不依赖于ProductCatalog的内部实现,  
        // 如果内部实现发生变化,客户端代码不需要修改。  
    }  
}

运行上面的代码会得到以下输出:

Product{name='Laptop', price=999.99}  
Product{name='Smartphone', price=499.99}  
Product{name='Tablet', price=299.99}

在这个例子中,ProductCatalog类实现了Iterable接口,并提供了一个iterator方法来创建并返回ProductIterator对象,ProductIteratorProductCatalog的一个私有内部类,它实现了Iterator接口来遍历产品列表,client代码使用增强的for循环(它内部使用了迭代器)来遍历ProductCatalog中的产品,而不需要知道其内部是如何存储产品的。

这个实现遵循了迭代器模式,允许client代码以统一的方式遍历聚合对象,而不需要了解聚合对象的内部表示,使得client代码与聚合对象之间实现了松耦合,符合开闭原则。

核心总结

程序员必知!迭代器模式的实战应用与案例分析 - 程序员古德

迭代器模式是一种常用的设计模式,它使程序能够遍历聚合对象的元素而无需暴露其内部表示,它支持对聚合对象的多种遍历,简化了聚合类,符合单一职责原则,使client代码与聚合对象的具体实现解耦,增强了系统的可扩展性,当需要新增遍历方式时,只需增加新的迭代器类,不影响原有代码;迭代器模式会涉及到较多的类,可能会增加系统的复杂性和开发成本,同时,由于它将存储数据和遍历数据的职分离,可能在某些情况下导致额外的性能开销,在需要遍历复杂数据结构或支持多种遍历方式时,推荐使用迭代器模式,但要注意权衡系统复杂性和灵活性,避免过度设计,同时,可以考虑使用Java内置的迭代器接口和工具类来简化实现。

关注我,每天学习互联网编程技术 - 程序员古德

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

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

相关文章

LeetCode-移动零(283)

题目描述&#xff1a; 给定一个数组 nums&#xff0c;编写一个函数将所有 0 移动到数组的末尾&#xff0c;同时保持非零元素的相对顺序。 请注意 &#xff0c;必须在不复制数组的情况下原地对数组进行操作。 思路&#xff1a; 这里的思路跟以前做过的去重复数字的思路有点像&…

整合事务,名词,概念

mysql实例(instance): MySQL是单进程多线程&#xff0c;也就是说MySQL实例在系 统上表现就是一个服务进程&#xff0c;即进程,就是我们在自己电脑布了一个mysql实例(服务),然后我们在代码中就可以通过配上mysql实例的地址,就能连上自己电脑上的这mysql实例. 数据库database: …

Vue框架底层

一、前端框架的由来 1、服务端渲染 sequenceDiagram 浏览器->>服务器: https://www.bilibili.com/ Note right of 服务器: 组装页面(服务端渲染) 服务器->>-浏览器: 完整页面2、前后端分离 sequenceDiagram 浏览器->>服务器: https://www.bilibili.com/ 服务…

Linux文件系统和日志分析

一、inode表结构 1. inode表 inode号在同一个设备上是唯一的。 inode号是有限资源&#xff0c;它的大小和磁盘大小有关。 访问文件的基本流程 根据文件夹的文件名和inode号的关系找到对应的inode表&#xff0c;再根据inode表&#xff08;属主 属组&#xff09;当中的指针找到磁…

毛概笔记。

一、 毛泽东思想是马中化的第一果&#xff0c;是关于搞革命&#xff0c;搞改造&#xff0c;搞建设的理论。 二、新民主主义革命 新民主主义革命的三大法宝&#xff1a;1.统一战线 2. 武装斗争 3.党的建设 政治纲领 经济纲领 文化纲领 乱世造英雄 三、社会主义改造理论&#xff…

多类指针式仪表自动读数系统的LabVIEW开发应用案例

多类指针式仪表自动读数系统的LabVIEW开发应用案例 工业环境中&#xff0c;多类指针式仪表的自动读数一直是一个具有挑战性的问题。本案例旨在展示如何使用LabVIEW开发一个高度智能化的多类指针式仪表自动读数系统&#xff0c;以应对复杂的工业环境。通过结合图像处理技术和深…

DSP2335的时钟PLL配置

PLL模块框图 xclkin是直接进来的外部时钟&#xff1b; 而下面的是振荡器&#xff08;晶振出来&#xff09;的时钟 PLLSTS 锁相环状态寄存器 PLLCR 锁相环控制寄存器 PLLSTS【oscoff】 决定着外部时钟的输入 PLLSTS【plloff】 锁相器关闭位 0使能PLL 锁相环控制寄存器…

程序猿的时间管理和生产力

文章目录 为什么时间管理很重要&#xff1f;如何管理时间&#xff1f;心理维度生理维度技术尺寸 时间管理技巧每周计划基于目标的规划番茄钟为什么是25分钟&#xff1f;番茄钟为什么有效&#xff1f;艾森豪威尔矩阵这一切都是从开发者的角度来看的 也许我从开始学习或从事软件开…

C++面向对象核心-权限-多态

1、权限 1.1 权限修饰符 三种权限&#xff0c;一共对应九种场景。要做到心中有表&#xff0c;遇到任何一种场景都能直接反映出是否能访问。 类内 派生类中 全局 private √ protected √ √ public √ √ √ #include <iostream> using namespace std;…

第二百五十三回

文章目录 概念介绍使用方法示例代码 概念介绍 进度条是常用的组件之一&#xff0c;它主要用来显示某种动作的完成进度。Flutter提供了多种进度条组件&#xff0c;常用的是水平进度条&#xff1a;LinearProgressIndicator&#xff1b;圆形进度条 :CircularProgressIndicator和R…

4.4 TILING FOR REDUCED MEMORY TRAFFIC

我们在CUDA中使用设备内存方面有一个内在的权衡&#xff1a;全局内存大但速度慢&#xff0c;而共享内存小但速度快。一个常见的策略是将数据划分为称为tile的子集&#xff0c;以便每个tile都适合共享内存。tile一词”借鉴了一个类比&#xff0c;即大墙&#xff08;即全局内存数…

data-factory java 开源根据对象定义自动生成测试对象数据

创作目的 我们平时在写测试用例的时候&#xff0c;免不了要写一大堆 set 方法为对象设置属性。 有时候为了补全测试用例&#xff0c;这件事就会变得非常枯燥。 于是就在想&#xff0c;能不能写一个可以自动生成测试对象的工具呢&#xff1f; 于是就有了这一个没啥用的测试框…

网络调试 UDP1,开发板用动态地址-入门6

https://www.bilibili.com/video/BV1zx411d7eC?p11&vd_source109fb20ee1f39e5212cd7a443a0286c5 1, 开发板连接路由器 1.1&#xff0c;烧录无OS UDP例程 1.2&#xff0c;Mini USB连接电脑 1.3&#xff0c;开发板LAN接口连接路由器 2. Ping开发板与电脑之间通信* 2.1 根据…

人工智能AI网站大全—实现自动聊天、绘画、创作论文、生成视频等

人工智能正在逐步改变大家的生活和工作方式&#xff0c;本文总结当前人工智能实用网站&#xff0c;方便大家更快地把AI应用到工作和生活中&#xff0c;提高效率。主要包括自动聊天、自动创作论文、自动绘画、、自动创作视频等模块。 文章目录 Part1 10w.aiPart2 liblib.AIPart3…

洗地机什么牌子好?目前口碑最好的洗地机

如今&#xff0c;人们的生活中&#xff0c;洗地机已经成为了越来越受欢迎的清洁工具&#xff0c;洗地机能迅速而有效地清理地板、地毯以及其他硬表面&#xff0c;为用户提供更加方便快捷的洗地机体验。那么&#xff0c;洗地机什么牌子好?我们一起来看看目前口碑最好的洗地机有…

人工智能任务2-读懂Transformer模型的十个灵魂拷问问题,深度理解Transformer模型架构

大家好&#xff0c;我是微学AI&#xff0c;今天给大家介绍一下人工智能任务2-读懂Transformer模型的十个灵魂拷问问题&#xff0c;深度理解Transformer模型架构。Transformer模型是一种基于自注意力机制的神经网络架构&#xff0c;被广泛用于自然语言处理任务中&#xff0c;如机…

C语言注意点(4)

1、void *a是什么意思 答&#xff1a;泛型指针&#xff0c;但不规定其类型(就是地址确定&#xff0c;但数据长度不确定)在动态分配内存时&#xff0c;malloc的返回值就是该类型&#xff0c;方便用户进行强制转换。 2、VS怎么一键规范格式 for(i0;i<10;i)enter后&#xff0c;…

selenium3自动化测试(这一篇就够了)——自学篇

本人整理收藏了20年多家公司面试知识点整理 &#xff0c;以及学习路线和视频教程免费分享给大家&#xff0c;我认为对面试来说是非常有用的&#xff0c;想要资料的话请点1150305204暗号CSDN。或者点击文末名片进入&#xff0c;免费领取 &#xff08;一&#xff09;安装seleniu…

120°AGV|RGV小车激光障碍物传感器|避障雷达DE系列安装与连线方法

120AGV|RGV小车激光障碍物传感器|避障雷达DE系列包含DE-4211、DE-4611、DE-4311、DE-4511等型号&#xff0c;根据激光飞行时间&#xff08;TOF&#xff09;测量原理运行的&#xff0c;利用激光光束对周围进行 120 半径 4m&#xff08;90%反射率&#xff09;扫描&#xff0c;获得…

一文初步了解slam技术

本文初步介绍slam技术&#xff0c;主要是slam技术的概述&#xff0c;涉及技术原理、应用场景、分类、以及各自优缺点&#xff0c;和slam技术的未来展望。 &#x1f3ac;个人简介&#xff1a;一个全栈工程师的升级之路&#xff01; &#x1f4cb;个人专栏&#xff1a;slam精进之…