【C++设计模式】第十六篇:迭代器模式(Iterator)

注意:复现代码时,确保 VS2022 使用 C++17/20 标准以支持现代特性。

遍历聚合对象的统一方式


1. 模式定义与用途

核心思想

  • 迭代器模式:提供一种方法顺序访问聚合对象的元素,而无需暴露其内部表示。
  • 关键用途
    1.​统一遍历接口:为不同数据结构(如数组、链表、树)提供一致的遍历方式。
    ​2.支持多种遍历策略:前向、反向、条件过滤等。
    ​3.简化聚合类设计:将遍历逻辑从聚合类中分离。

经典场景

  • STL容器的迭代器(如std::vector::iterator)。
  • 自定义集合类(如链表、图)的遍历。
  • 数据库查询结果的逐行遍历。

2. 模式结构解析

UML类图

+---------------------+          +---------------------+  
|      Aggregate      |          |       Iterator       |  
+---------------------+          +---------------------+  
| + createIterator()  |<>------->| + next(): void       |  
+---------------------+          | + hasNext(): bool    |  
          ^                      +---------------------+  
          |                                ^  
          |                                |  
  +-------+-------+              +---------+---------+  
  |               |              |                   |  
+---------------------+    +-------------------+ +----------------+  
| ConcreteAggregate   |    |  ConcreteIterator | |     Client      |  
+---------------------+    +-------------------+ +----------------+  
| + createIterator()  |    | + next()          | | 通过迭代器遍历聚合对象 |  
+---------------------+    | + hasNext()       | +----------------+  
                           +-------------------+  

角色说明

  1. Aggregate:聚合接口,定义创建迭代器的方法(如createIterator())。
  2. ConcreteAggregate:具体聚合类,实现迭代器创建逻辑。
  3. Iterator:迭代器接口,定义遍历方法(如next()hasNext())。
  4. ConcreteIterator:具体迭代器,实现特定遍历逻辑。
  5. Client:通过迭代器访问聚合对象,无需依赖其内部结构。

3. 现代C++实现示例

场景:自定义链表迭代器

步骤1:定义链表节点与聚合类
#include <iostream>  
#include <memory>  

template <typename T>  
class ListNode {  
public:  
    T value;  
    std::shared_ptr<ListNode<T>> next;  
    ListNode(T val) : value(val), next(nullptr) {}  
};  

// 聚合类:单向链表  
template <typename T>  
class LinkedList {  
public:  
    void append(T value) {  
        auto newNode = std::make_shared<ListNode<T>>(value);  
        if (!head_) {  
            head_ = newNode;  
        } else {  
            tail_->next = newNode;  
        }  
        tail_ = newNode;  
    }  

    // 创建正向迭代器  
    class Iterator;  
    Iterator begin() { return Iterator(head_); }  
    Iterator end() { return Iterator(nullptr); }  

private:  
    std::shared_ptr<ListNode<T>> head_ = nullptr;  
    std::shared_ptr<ListNode<T>> tail_ = nullptr;  
};  
步骤2:实现迭代器类
template <typename T>  
class LinkedList<T>::Iterator {  
public:  
    Iterator(std::shared_ptr<ListNode<T>> node) : current_(node) {}  

    T& operator*() const { return current_->value; }  
    Iterator& operator++() {  
        if (current_) current_ = current_->next;  
        return *this;  
    }  
    bool operator!=(const Iterator& other) const {  
        return current_ != other.current_;  
    }  

private:  
    std::shared_ptr<ListNode<T>> current_;  
};  
步骤3:客户端代码
int main() {  
    LinkedList<int> list;  
    list.append(1);  
    list.append(2);  
    list.append(3);  

    // 使用范围for循环(依赖begin()和end())  
    for (auto num : list) {  
        std::cout << num << " ";  // 输出:1 2 3  
    }  

    // 手动迭代  
    auto it = list.begin();  
    while (it != list.end()) {  
        std::cout << *it << " ";  
        ++it;  
    }  
}  
扩展:反向迭代器
template <typename T>  
class LinkedList<T>::ReverseIterator {  
public:  
    ReverseIterator(std::shared_ptr<ListNode<T>> head) {  
        // 遍历链表,将节点指针存入栈以实现反向  
        auto curr = head;  
        while (curr) {  
            stack_.push(curr);  
            curr = curr->next;  
        }  
    }  

    T& operator*() { return stack_.top()->value; }  
    ReverseIterator& operator++() {  
        if (!stack_.empty()) stack_.pop();  
        return *this;  
    }  
    bool operator!=(const ReverseIterator& other) {  
        return !stack_.empty() || !other.stack_.empty();  
    }  

private:  
    std::stack<std::shared_ptr<ListNode<T>>> stack_;  
};  

4. 应用场景示例

场景1:树结构的深度优先遍历

class TreeNode {  
public:  
    int value;  
    std::vector<std::shared_ptr<TreeNode>> children;  
};  

class DepthFirstIterator {  
public:  
    DepthFirstIterator(std::shared_ptr<TreeNode> root) {  
        stack_.push(root);  
    }  

    std::shared_ptr<TreeNode> next() {  
        auto node = stack_.top();  
        stack_.pop();  
        for (auto it = node->children.rbegin(); it != node->children.rend(); ++it) {  
            stack_.push(*it);  
        }  
        return node;  
    }  

    bool hasNext() { return !stack_.empty(); }  

private:  
    std::stack<std::shared_ptr<TreeNode>> stack_;  
};  

场景2:过滤迭代器(条件遍历)

template <typename T, typename Predicate>  
class FilterIterator {  
public:  
    FilterIterator(typename LinkedList<T>::Iterator it, Predicate pred)  
        : it_(it), pred_(pred) {  
        // 找到第一个满足条件的元素  
        while (it_ != end_ && !pred_(*it_)) ++it_;  
    }  

    T& operator*() { return *it_; }  
    FilterIterator& operator++() {  
        do { ++it_; } while (it_ != end_ && !pred_(*it_));  
        return *this;  
    }  
    bool operator!=(const FilterIterator& other) { return it_ != other.it_; }  

private:  
    typename LinkedList<T>::Iterator it_;  
    typename LinkedList<T>::Iterator end_;  
    Predicate pred_;  
};  

// 使用示例:遍历链表中的偶数  
auto isEven = [](int x) { return x % 2 == 0; };  
FilterIterator<int, decltype(isEven)> begin(list.begin(), isEven);  
FilterIterator<int, decltype(isEven)> end(list.end(), isEven);  
while (begin != end) {  
    std::cout << *begin << " ";  
    ++begin;  
}  

5. 优缺点分析

​优点​缺点
解耦遍历逻辑与数据结构增加类的数量(迭代器与聚合类需配对)
支持多种遍历策略(正向、反向等)复杂数据结构迭代器实现成本高(如图遍历)
隐藏聚合对象内部实现部分语言/框架已内置迭代器(如STL)

6. 调试与优化策略

调试技巧(VS2022)​

1.​验证迭代器有效性:
  • 在迭代器越界时触发断言:
T& operator*() {  
    assert(current_ != nullptr && "迭代器越界!");  
    return current_->value;  
}  
2. ​检查迭代器状态:
  • operator++()中设置断点,观察指针移动是否符合预期。

性能优化

1. 预计算遍历路径:
  • 对树或图的遍历,预计算路径并缓存结果(如广度优先遍历队列)。
2. 内存连续性优化:
  • 使用std::vector存储节点,利用内存局部性提升遍历速度。

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

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

相关文章

手机屏幕摔不显示了,如何用其他屏幕临时显示,用来导出资料或者清理手机

首先准备一个拓展坞 然后 插入一个外接的U盘 插入鼠标 插入有数字小键盘区的键盘 然后准备一根高清线&#xff0c;一端链接电脑显示器,一端插入拓展坞 把拓展坞的连接线&#xff0c;插入手机充电口&#xff08;可能会需要转接头&#xff09; 然后确保手机开机 按下键盘…

基于SpringBoot的“文物管理系统”的设计与实现(源码+数据库+文档+PPT)

基于SpringBoot的“文物管理系统”的设计与实现&#xff08;源码数据库文档PPT) 开发语言&#xff1a;Java 数据库&#xff1a;MySQL 技术&#xff1a;SpringBoot 工具&#xff1a;IDEA/Ecilpse、Navicat、Maven 系统展示 系统总体功能模块图 E-R实体图 系统首页界面 系统…

微信小程序投票系统的构建与实现

在数字化时代&#xff0c;微信小程序已经成为人们日常生活的重要组成部分。无论是企业宣传、活动组织还是社交互动&#xff0c;小程序都展现出了其强大的功能和便捷的用户体验。其中&#xff0c;微信小程序的投票系统尤为突出&#xff0c;它不仅能简化投票流程&#xff0c;还能…

【Java篇】数据类型与变量:窥见程序的天地万象

文章目录 Java 数据类型与变量基础&#xff1a;从零开始&#xff0c;轻松掌握前言一、字面常量1.1 什么是字面常量&#xff1f;1.2 字面常量的分类1.2.1 字符串常量1.2.2 整形常量1.2.3 浮点数常量1.2.4 字符常量1.2.5 布尔常量1.2.6 空常量 1.3 字面常量与数据类型 二、Java 中…

绿盟春招面试题

《网安面试指南》https://mp.weixin.qq.com/s/RIVYDmxI9g_TgGrpbdDKtA?token1860256701&langzh_CN 5000篇网安资料库https://mp.weixin.qq.com/s?__bizMzkwNjY1Mzc0Nw&mid2247486065&idx2&snb30ade8200e842743339d428f414475e&chksmc0e4732df793fa3bf39…

操作系统控制台-健康守护我们的系统

引言基本准备体验功能健康守护系统诊断 收获提升结语 引言 阿里云操作系统控制平台作为新一代云端服务器中枢平台&#xff0c;通过创新交互模式重构主机管理体验。操作系统控制台提供了一系列管理功能&#xff0c;包括运维监控、智能助手、扩展插件管理以及订阅服务等。用户可以…

Linux系统基于ARM平台的LVGL移植

软硬件介绍&#xff1a;Ubuntu 20.04 ARM 和&#xff08;Cortex-A53架构&#xff09;开发板 基本原理 LVGL图形库是支持使用Linux系统的Framebuffer帧缓冲设备实现的&#xff0c;如果想要实现在ARM开发板上运行LVGL图形库&#xff0c;那么就需要把LVGL图形库提供的关于帧缓冲设…

DIY Tomcat:手写一个简易Servlet容器

在Java Web开发领域&#xff0c;Tomcat堪称经典&#xff0c;它作为Servlet容器&#xff0c;承载着无数Web应用的运行。今天&#xff0c;我将带大家一同探索如何手写一个简易的Tomcat&#xff0c;深入理解其底层原理。 一、背景知识 在开始之前&#xff0c;我们需要对几个关键…

PDF转JPG(并去除多余的白边)

首先&#xff0c;手动下载一个软件&#xff08;poppler for Windows&#xff09;&#xff0c;下载地址&#xff1a;https://github.com/oschwartz10612/poppler-windows/releases/tag/v24.08.0-0 否则会出现以下错误&#xff1a; PDFInfoNotInstalledError: Unable to get pag…

网络安全之端口扫描(一)

前置介绍 什么是DVWA&#xff1f; DVWA&#xff08;Damn Vulnerable Web Application&#xff09;是一个专门设计用于测试和提高Web应用程序安全技能的开源PHP/MySQL Web应用程序。它是一个具有多个安全漏洞的故意不安全的应用程序&#xff0c;供安全专业人员、渗透测试人员、…

财务会计域——合并报表系统设计

摘要 本文主要介绍了合并报表系统的设计&#xff0c;包括其背景、业务流程和系统架构设计。合并报表系统可自动化生成数据&#xff0c;减少人为错误&#xff0c;确保报表合规。其业务流程涵盖数据收集、标准化、合并调整、报表生成、审核及披露等环节。系统架构设计包括数据接…

游戏引擎学习第147天

仓库:https://gitee.com/mrxiao_com/2d_game_3 上一集回顾 具体来说&#xff0c;我们通过隐式计算来解决问题&#xff0c;而不是像数字微分分析器那样逐步增加数据。我们已经涵盖了这个部分&#xff0c;并计划继续处理音量问题。不过&#xff0c;实际上我们现在不需要继续处理…

NoSQL数据库系统Cassandra学习笔记

详细文档&#xff1a;我用夸克网盘分享了「noSQL.pdf」&#xff0c;点击链接即可保存。打开「夸克APP」在线查看&#xff0c;支持多种文档格式转换。 链接&#xff1a;https://pan.quark.cn/s/dfc3864807b4 参考链接&#xff1a;黑马程序员NoSQL数据库系统Cassandra全套教程&a…

苹果 M3 Ultra 芯片深度解析:AI 时代的性能革命

2025 年 3 月 5 日&#xff0c;苹果正式发布了其史上最强 PC 芯片 ——M3 Ultra。这款基于 UltraFusion 封装技术的旗舰级 SoC&#xff0c;不仅延续了苹果芯片在能效比上的传统优势&#xff0c;更通过架构创新与硬件升级&#xff0c;将 AI 计算能力推向了新高度。本文将从性能突…

Qt从入门到入土(八) -打包Qt程序

前言 当你写完一个有趣的Qt程序时&#xff0c;想发给朋友或者家人&#xff0c;但是他们的电脑又没有安装Qt&#xff0c;那么如何直接在他们电脑上运行又不需要安装Qt呢&#xff1f;本篇文章会告诉你答案&#xff0c;本文详细的介绍了界面设计和功能实现。读完本文你不仅可以学…

使用OpenCV和MediaPipe库——实现人体姿态检测

目录 准备工作如何在Windows系统中安装OpenCV和MediaPipe库&#xff1f; 安装Python 安装OpenCV 安装MediaPipe 验证安装 代码逻辑 整体代码 效果展示 准备工作如何在Windows系统中安装OpenCV和MediaPipe库&#xff1f; 安装Python 可以通过命令行运行python --versio…

React:Axios

axios可以在浏览器和node.js两边跑&#xff0c;可以向服务端发起ajax请求&#xff0c;也可以在node.js里运行&#xff0c;向远端服务发送http请求 Axios中文文档 | Axios中文网 <!DOCTYPE html> <html lang"en"> <head><meta charset"UT…

数据结构第八节:红黑树(初阶)

【本节要点】 红黑树概念红黑树性质红黑树结点定义红黑树结构红黑树插入操作的分析 一、红黑树的概念与性质 1.1 红黑树的概念 红黑树 &#xff0c;是一种 二叉搜索树 &#xff0c;但 在每个结点上增加一个存储位表示结点的颜色&#xff0c;可以是 Red和 Black 。 通过对 任何…

使用 vxe-table 导出 excel,支持带数值、货币、图片等带格式导出

使用 vxe-table 导出 excel&#xff0c;支持带数值、货币、图片等带格式导出&#xff0c;通过官方自动的导出插件 plugin-export-xlsx 实现导出功能 查看官网&#xff1a;https://vxetable.cn gitbub&#xff1a;https://github.com/x-extends/vxe-table gitee&#xff1a;htt…

C# Unity 唐老狮 No.7 模拟面试题

本文章不作任何商业用途 仅作学习与交流 安利唐老狮与其他老师合作的网站,内有大量免费资源和优质付费资源,我入门就是看唐老师的课程 打好坚实的基础非常非常重要: 全部 - 游习堂 - 唐老狮创立的游戏开发在线学习平台 - Powered By EduSoho 如果你发现了文章内特殊的字体格式,…