新闻中心

C++怎么实现一个访问者模式(Visitor)_C++设计模式与访问者模式实现

2025-11-20
浏览次数:
返回列表
访问者模式通过双重分发在不修改元素类的情况下添加新操作,由Element、ConcreteElement、Visitor、ConcreteVisitor和ObjectStructure组成,适用于数据结构稳定而操作多变的场景。

c++怎么实现一个访问者模式(visitor)_c++设计模式与访问者模式实现

访问者模式(Visitor Pattern)是一种行为设计模式,它允许你在不修改对象结构的前提下,为对象结构中的元素添加新的操作。C++中实现访问者模式的关键是“双重分发”——通过虚函数实现动态绑定,把操作从数据结构中解耦出来。

访问者模式的核心组成

访问者模式包含以下几个关键角色:

  • Element(元素):定义一个接受访问者的接口,比如 accept(Visitor&)
  • ConcreteElement:具体元素类,实现 accept 方法,将调用转发给访问者的 visit 方法。
  • Visitor(访问者):定义对每种元素的访问方法,如 visit(ConcreteElementA&)
  • ConcreteVisitor:具体访问者,实现各种 visit 方法,封装具体操作。
  • ObjectStructure:能枚举元素并让访问者遍历它们的结构(比如容器)。

基本实现代码示例

下面是一个简单的 C++ 实现,展示如何使用访问者模式处理不同类型的文档元素(文本和图片):

#include <iostream>
#include <vector>
#include <memory>

// 前向声明
class TextElement;
class ImageElement;

// 访问者接口
class Visitor {
public:
    virtual ~Visitor() = default;
    virtual void visit(TextElement& element) = 0;
    virtual void visit(ImageElement& element) = 0;
};

// 元素基类
class Element {
public:
    virtual ~Element() = default;
    virtual void accept(Visitor& visitor) = 0;
};

// 具体元素:文本
class TextElement : public Element {
public:
    void accept(Visitor& visitor) override {
        visitor.visit(*this);
    }

    std::string getText() const {
        return "This is a text block.";
    }
};

// 具体元素:图片
class ImageElement : public Element {
public:
    void accept(Visitor& visitor) override {
        visitor.visit(*this);
    }

    std::string getImageInfo() const {
        return "Image: photo.jpg, size 1920x1080";
    }
};

// 具体访问者:打印访问者
class PrintVisitor : public Visitor {
public:
    void visit(TextElement& element) override {
        std::cout << "[Print] " << element.getText() << "\n";
    }

    void visit(ImageElement& element) override {
        std::cout << "[Print] " << element.getImageInfo() << "\n";
    }
};

// 具体访问者:导出为 HTML
class HtmlExportVisitor : public Visitor {
public:
    void visit(TextElement& element) override {
        std::cout << "<p>" << element.getText() << "</p>\n";
    }

    void visit(ImageElement& element) override {
        std::cout << "@@##@@\n"; // 简化提取文件名
    }
};

// 文档结构(对象结构)
class Document {
private:
    std::vector<std::unique_ptr<Element>> elements;

public:
    void addElement(std::unique_ptr<Element> element) {
        elements.push_back(std::move(element));
    }

    void accept(Visitor& visitor) {
        for (auto& e : elements) {
            e->accept(visitor);
        }
    }
};

使用方式

你可以轻松地添加新操作而无需改动元素类:

int main() {
    Document doc;
    doc.addElement(std::make_unique<TextElement>());
    doc.addElement(std::make_unique<ImageElement>());

    PrintVisitor printVisitor;
    HtmlExportVisitor htmlVisitor;

    std::cout << "=== 打印文档 ===\n";
    doc.accept(printVisitor);

    std::cout << "\n=== 导出为 HTML ===\n";
    doc.accept(htmlVisitor);

    return 0;
}

输出结果:

小云雀 小云雀

剪映出品的AI视频和图片创作助手

小云雀 1949 查看详情 小云雀
=== 打印文档 ===
[Print] This is a text block.
[Print] Image: photo.jpg, size 1920x1080
<p>=== 导出为 HTML ===</p><p>This is a text block.</p>@@##@@

访问者模式的优缺点

优点:

  • 新增操作非常方便,只需添加新的访问者类。
  • 符合开闭原则:对扩展开放,对修改关闭。
  • 集中相关操作,避免污染元素类。

缺点:

  • 如果元素类型频繁变化,每个访问者都要修改,违背开闭原则。
  • 代码可读性降低,尤其是初学者不易理解双重分发机制。
  • C++ 中需要前向声明和多重指针管理,稍显繁琐。

基本上就这些。访问者模式适合数据结构稳定但操作多变的场景,比如编译器 AST 处理、文档渲染、序列化等。用好它,能让系统更灵活。

imageC++怎么实现一个访问者模式(Visitor)_C++设计模式与访问者模式实现

以上就是C++怎么实现一个访问者模式(Visitor)_C++设计模式与访问者模式实现的详细内容,更多请关注其它相关文章!


# 头文件  # 关键词排名会有波动吗  # 证券的推广营销  # 推广水果的营销手段  # 甘肃省教育行业营销推广  # 低价网站建设费用预算  # 网站大图片优化  # 谷歌seo月流量百万  # 营销推广客群怎么找  # 做seo如何赚钱  # 肇庆网站建设免费咨询  # 是一种  # 开闭  # 几个  # c++设计模式  # 是一个  # 前向  # 如何实现  # 如何使用  # 文档  # 数据结构  # 代码可读性  # stream  # ios  # c++  # ai  # html  # 访问者模式 


相关栏目: 【 科技资讯46185 】 【 网络学院92790


相关推荐: poki网页游戏推荐_poki免费游戏平台入口  J*aScript生成器_j*ascript异步迭代  J*aScript井字棋(Tic-Tac-Toe)核心交互逻辑实现教程  谷歌浏览器无痕模式怎么开 Chrome开启无痕浏览设置方法【教程】  Python中如何避免重复条件判断:利用数据结构实现动态逻辑  2026年发布! 美少女养成动作RPG《神剑少女战记》发布实机演示  蛙漫安全无毒 官方认证的绿色入口  晋江读书网页版在线登录 晋江读书电脑版官网  AO3官方在线访问地址 Archive of Our Own最新镜像合集  2025年云电脑操作系统体验 | 无需本地硬件,随时随地使用高性能PC  学习通网页版快速入口 学习通官网网页版直接打开  Lar*el DB::listen 事件中的查询执行时间单位解析  Win11怎么开启高性能模式_Windows 11电源计划优化设置  俄罗斯搜索引擎Yandex指南 附2025年免登录官网入口  126邮箱账号注册 电脑版登录入口  sublime怎么设置启动时打开的窗口_sublime会话管理与热退出  快手极速版在线观看 官方网页版登录地址  百度浏览器字体显示异常偏小_百度浏览器字体渲染修复方案  Yandex官方入口网址 Yandex俄罗斯搜索引擎最新在线地址  在Go Martini框架中高效服务动态生成图像的实践指南  LINQ to XML为何解析失败? 深入理解C# XDocument的异常处理  怎么在浏览器上运行HTML文件_浏览器运行HTML文件技巧【技巧】  Golang如何使用context实现超时取消_Golang context超时取消模式实践  uc手机浏览器网页版入口 uc浏览器手机版便捷登录首页  邮政快递单号查询入口 邮政快递物流信息在线查询入口  如何在Promise链中优雅地中断后续then执行  J*aScript类型检查_j*ascript代码规范  蓝湖怎样用切图标注提对接效率_蓝湖用切图标注提对接效率【设计对接】  小红书网页版入口链接分享 小红书官网直接进  sublime如何处理大型CSV文件的列对齐_sublime高级表格编辑插件指南  iwriter统一登录平台 iwrite账号密码登录页面  JUnit5/Mockito:优雅测试内部依赖与异常处理的实践  qq游戏免费畅玩入口_qq游戏电脑版快速启动  c++ 获取系统当前时间 c++时间戳获取方法  J*a应用程序首次运行自动创建文件与目录的最佳实践  j*a toString()的覆盖  sublime怎么格式化代码_sublime代码美化与一键排版插件配置  手机CPU怎么影响游戏体验_手机CPU对游戏性能的影响分析  CSS Flexbox如何实现多行排列_flex-wrap wrap自动换行显示  Golang如何优雅处理error_Golang error处理最佳实践总结  抖音网页版快捷访问 抖音网页版网页版入口操作教程  如何使用J*aScript精确选择并批量修改特定父元素下子链接的样式  小红书怎么解除第三方平台绑定_小红书多平台登录解绑方法介绍  Golang如何实现容器化日志收集与分析_Golang容器日志收集分析方法  在Go语言中利用后缀数组处理多字符串:实现高效文本匹配与自动补全  微信客户端如何收红包_微信客户端接收红包使用教程  Yandex搜索引擎官网入口_俄罗斯Yandex免登录一键直达  神经网络二分类模型训练异常:高损失与完美验证准确率的排查与修正  快手网页版在线登录 快手网页版官网入口快速访问  黑鲨3Pro怎样在相册开漫画风滤镜_iPhone黑鲨3Pro相册开漫画风滤镜【趣味滤镜】 

搜索