迭代器模式-C++实现

题外话:
设计模式是在1994年提出的,当时还没有C++的STL库和泛型编程,所以人们为了提供一种有效的方法来访问一个聚合对象(例如列表、集合、数组等)中的元素,而又不暴露该对象的内部表示,于是想到了迭代器模式。

但随着1998年STL库的诞生,我们知道STL里的容器都是支持迭代器遍历,在这样的情况下我们再使用传统的迭代器模式去遍历元素效率就会很慢。

因为传统的迭代器模式是多态的形式,也就是运行时多态,运行时多态我们需要去访问虚函数表找到要执行的函数地址,在这种情况下当我们的元素有一百万个时,遍历元素就会很慢,完全没必要。

但是虽然设计已经过时,思想却值得我们去学习。

迭代器模式是一种行为型设计模式,它提供了一种有效的方法来访问一个聚合对象(例如列表、集合、数组等)的元素,而不暴露该对象的内部表示。

迭代器模式包含以下四个角色:

1、抽象迭代器:定义迭代聚合对象的接口,提供方法用于获取下一个元素、判断是否还有元素等。

2、具体迭代器:实现抽象迭代器的接口,负责管理迭代的状态,例如当前位置、下一个元素等。

3、抽象聚合对象:定义一个迭代器工厂方法,用于创建一个具体迭代器的实例。

3、具体聚合对象:实现聚合对象接口,返回一个具体迭代器实例。

举例:

#include <iostream>
#include <memory>
#include <vector>// 抽象迭代器
template<class T>
class Iterator
{
public:virtual ~Iterator() {}// 获取下一个元素virtual T Next() = 0;// 判断是否还有元素virtual bool Empty() = 0;
};// 具体迭代器
template<class T>
class ConcreteIterator: public Iterator<T>
{
public:ConcreteIterator(const std::vector<T>& _col){idx_ = 0;data_ = _col;}virtual T Next() override{return data_[idx_++];}virtual bool Empty() override{return idx_ >= data_.size();}private:// 当前位置int idx_;// 数据std::vector<T> data_;
};// 抽象聚合对象
template<class T>
class Aggregate
{
public:virtual ~Aggregate() {}virtual void PushBack(const T& _val) = 0;virtual std::shared_ptr<Iterator<T>> CreateIterator() = 0;
};// 具体聚合对象
template<class T>
class ConcreteAggregate: public Aggregate<T>
{
public:virtual void PushBack(const T& _val) override{data_.emplace_back(_val);}virtual std::shared_ptr<Iterator<T>> CreateIterator() override{return std::make_shared<ConcreteIterator<T>>(data_);}private:std::vector<T> data_;
};

在代码中,我们定义了一个Iterator接口,其中包含了Next()和Empty()方法,分别用于获取下一个元素的数据和判断是否还有下一个元素。然后我们实现了一个ConcreteIterator类作为具迭代器的具体实现,它使用了一个std::vector作为聚合对象的集合,并通过追踪当前位置来实现迭代功能。

接下来,我们定义了一个Aggregate接口,其中包含了PushBack()和CreateIterator方法,用于插入元素和创建迭代器。然后实现了一个ConcreteAggregate类作为具体聚合对象的实现,它使用一个std::vector作为数据集合,并且在CreateIterator()方法中返回了一个具体迭代器对象。

测试:

void TestIterator()
{// 创建一个聚合对象std::shared_ptr<Aggregate<int>> aggregate = std::make_shared<ConcreteAggregate<int>>();// 插入数据aggregate->PushBack(1);aggregate->PushBack(2);aggregate->PushBack(3);aggregate->PushBack(4);aggregate->PushBack(5);// 创建迭代器std::shared_ptr<Iterator<int>> iterator = aggregate->CreateIterator();// 遍历元素输出while (!iterator->Empty())std::cout << iterator->Next() << " ";std::cout << std::endl;
}

测试代码中,我们创建一个聚合对象,并插入了5个元素,然后又创建了一个迭代器的对象,通过迭代器遍历元素并输出。

输出结果:

1 2 3 4 5

从上面的例子我们也能看出,迭代器模式其实和STL中的迭代器是一模一样的,不一样的是泛型编程中迭代器是编译时多态,而迭代器模式下是运行时多态,迭代器模式下遍历元素调用的函数需要通过虚函数表去查找函数地址,当元素太多时消耗的资源太多,对现在的C++编程来说已经不适用了。

总结:

1、迭代抽象:访问一个聚合对象的内容而无需暴露它的内部展示。

2、迭代多态:为遍历不同的集合结构提供一个统一的接口,从而支持同样的算法在不同的集合结构上进行操作。

3、迭代器的健壮性考虑:遍历的同时更改迭代器所在的集合结构,会导致问题。

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

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

相关文章

Pandas在Excel同一个sheet里插入多个Dataframe和行

Pandas默认的to_excel是直接把完成的Datafrme写入一个sheet里,这并不能满足我们在一个sheet里插入多个Dataframe或多行的需求。为了实现插入多行或多Dataframe的目的,我们需要新建一个ExcelWriter对象,然后依次插入数据。 这里我们以插入2个Dataframe和三行单元格为例。 新…

大华技术GIS开发工程师24届秋招三场面试Offer面经

本文介绍2024届秋招中&#xff0c;大华技术股份有限公司的GIS开发工程师岗位的3场面试基本情况、提问问题等。 10月投递了大华技术股份有限公司的GIS开发工程师岗位&#xff0c;所在部门为研发中心。目前完成了一面、二面与三面等全部流程&#xff0c;并有幸获得Offer&#xff…

vite脚手架,手写实现配置动态生成路由

参考文档 vite的glob-import vue路由配置基本都是重复的代码&#xff0c;每次都写一遍挺难受&#xff0c;加个页面就带配置下路由 那就利用 vite 的 文件系统处理啊 先看实现效果 1. 考虑怎么约定路由&#xff0c;即一个文件夹下&#xff0c;又有组件&#xff0c;又有页面&am…

[数据结构]HashSet与LinkedHashSet的底层原理学习心得

我们区分list和set集合的标准是三个&#xff1a;有无顺序&#xff0c;可否重复&#xff0c;有无索引。 list的答案是&#xff1a;有顺序&#xff0c;可重复&#xff0c;有索引。这也就是ArrayList和LinkedList的共性 set的答案是&#xff1a;顺序内部再区分,不可以重复&#xf…

【数电笔记】11-最小项(逻辑函数的表示方法及其转换)

目录 说明&#xff1a; 逻辑函数的建立 1. 分析逻辑问题&#xff0c;建立逻辑函数的真值表 2. 根据真值表写出逻辑式 3. 画逻辑图 逻辑函数的表示 1. 逻辑表达式的常见表示形式与转换 2. 逻辑函数的标准表达式 &#xff08;1&#xff09;最小项的定义 &#xff08;2&am…

python-迭代器与生成器

迭代器&#xff08;Iterators&#xff09;和生成器&#xff08;Generators&#xff09;是 Python 中用于处理可迭代对象的重要工具。它们在处理大型数据集或需要逐个产生元素的情况下非常有用。下面是关于这两个概念的中文介绍&#xff1a; 迭代器&#xff08;Iterators&#…

【JavaEE】多线程(3) -- 线程等待 wait 和 notify

目录 1. wait()⽅法 2. notify()⽅法 3. notifyAll()⽅法 4. wait 和 sleep 的对⽐&#xff08;⾯试题&#xff09; 由于线程之间是抢占式执⾏的, 因此线程之间执⾏的先后顺序难以预知. 但是实际开发中有时候我们希望合理的协调多个线程之间的执⾏先后顺序. 完成这个协调⼯…

洛谷 P1450 [HAOI2008] 硬币购物

思路 完全背包&#xff1a;预处理出不限制硬币数量的方案数。 dp[0]1; dfor(i,1,4) dfor(j,c[i],(int)1e5) dp[j]dp[j-c[i]];容斥 不限制数量的方案数 − - − 超出限制的方案数 符合限制的方案数 。考虑第 i i i 种硬币超出数量限制的方案数。强制支付 d i 1 d_i1 di​…

Ubuntu18.04的VSCode输入不了中文

在Ubuntu使用VSCode写代码想写注释的&#xff0c;但是发现不能输入中文&#xff0c;上网寻找解决方案&#xff0c;这个方法也是可以的&#xff08;根据2的命令操作就可以了&#xff0c;2也使参考1的&#xff09;&#xff1a; Visual Studio Code on Linux-Debian and Ubuntu b…

【数电笔记】18-卡诺图化简

目录 说明&#xff1a; 用卡诺图化简逻辑函数 1. 公式法化简与卡诺图化简对比 2. 化简依据 3. 化简规律 3.1 两个小方块相邻 3.2 四个小方块相邻 3.3 八个小方块相邻 4. 卡诺图化简法步骤 4.1 例1 4.2 例2 5. 画卡诺圈规则 5.1 例1 6. 特殊情况 6.1 例1 6.2 例…

【JVM】一篇通关JVM类加载与字节码技术

目录 1. 类文件结构1-1. 魔数 版本 常量池 2. 字节码指令3. 编译期处理4. 类加载阶段5. 类加载器6. 运行期优化 类加载与字节码技术 1. 类文件结构 案例 // HelloWorld 示例 public class HelloWorld {public static void main(String[] args) {System.out.println("h…

职责链模式-C++实现

职责链模式是一种行为型设计模式&#xff0c;它允许多个对象以链式的方式处理请求&#xff0c;避免了请求的发送者和接收者之间的耦合关系。将这些对象连成一条链&#xff0c;并沿着这条链传递请求&#xff0c;直到有一个对象处理请求为止&#xff0c;也就是说如果当前对象不处…

[Linux] linux防火墙

一、防火墙是什么 防火墙&#xff08;FireWall&#xff09;&#xff1a;隔离功能&#xff0c;工作在网络或主机的边缘&#xff0c;数据包的匹配规则与由一组功能定义的操作组件处理的规则相匹配&#xff0c;根据特定规则检查网络或主机的入口和出口 当要这样做时&#xff0c;基…

shell编程-sed命令详解(超详细)

文章目录 前言一、sed命令介绍1. sed命令简介2. sed命令的基本语法3. 常用的sed命令选项&#xff1a;4. 常用的sed编辑命令&#xff1a; 二、sed命令示例用法1. 替换文本2. 删除匹配行&#xff1a;3. 打印特定行&#xff1a;4. 在指定行之前插入文本&#xff1a;5. 从文件中读取…

springboot(ssm stone音乐播放器 在线音乐平台Java(codeLW)

springboot(ssm stone音乐播放器 在线音乐平台Java(code&LW) 开发语言&#xff1a;Java 框架&#xff1a;ssm/springboot vue JDK版本&#xff1a;JDK1.8&#xff08;或11&#xff09; 服务器&#xff1a;tomcat 数据库&#xff1a;mysql 5.7&#xff08;或8.0&#x…

C++函数模板,类模板

C函数模板&#xff0c;类模板 1.函数模板1.1函数模板的概念1.2函数模板的格式1.3函数模板的原理1.4函数模板的实例化1.5模板参数的匹配原则 2.类模板2.1类模板的定义格式2.2类模板的实例化 1.函数模板 1.1函数模板的概念 在C中&#xff0c;函数模板是一种通用的函数定义&…

Mybatis缓存机制

系列文章目录 1、mybatis简介及数据库连接池 2、mybatis简单使用 3、mybatis中selectOne的使用 4、mybatis中resultMap结果集的使用 5、mybatis实用教程之XML实现动态sql 6、Mybatis使用注解实现复杂动态SQL Mybatis缓存机制 系列文章目录前言1、配置二级缓存2、使用场景&…

Linux系统-----进程通讯

前言 本期我们来学习进程间的通讯 一、信号机制 1、信号的基本概念 每个信号都对应一个正整数常量(称为signal number,即信号编号。定义在系统头文件<signal.h>中)&#xff0c;代表同一用户的诸进程之间传送事先约定的信息的类型&#xff0c;用于通知某进程发生了某异常…

Java 使用对应arthas 调试程序

1、作用 使用 arthas 可以进行如下操作 ① 抓取对应函数的耗时结构&#xff0c;然后分析对应的代码优化代码 ② 抓取对应函数的 入参、出参函数 ③ 重放对应的函数执行 ④ 查询对应程序占用结构&#xff0c;比如 cpu, jvm ⑤ 查询对应的 执行最频繁的 线程 ⑥ 打印函数…

开源软件license介绍与检测

开源License介绍 通俗来讲&#xff0c;开源许可证就是一种允许软件使用者在一定条件内按照需要自由使用和修改软件及其源代码的的法律条款。借此条款&#xff0c;软件作者可以将这些权利许可给使用者&#xff0c;并告知使用限制。这些许可条款可以由个人、商业公司或非赢利组织…