0

0

c++ 访问者模式实现 c++如何对对象结构中的元素进行操作

幻夢星雲

幻夢星雲

发布时间:2026-03-17 14:07:33

|

247人浏览过

|

来源于php中文网

原创

访问者模式的核心是双分派,用于对象结构稳定但需频繁添加新操作的场景;它通过accept()和visit()两次虚调用模拟双分派,要求accept()为虚函数且各节点重写,visit()按具体类型重载并统一用const T&参数,避免const错误与状态污染。

c++ 访问者模式实现 c++如何对对象结构中的元素进行操作

访问者模式的核心是「双分派」,不是继承多态

它解决的问题很具体:当对象结构稳定(比如 ASTNode 类型树不变),但需要频繁添加新操作(比如打印、类型检查、代码生成)时,避免反复修改每个节点类。C++ 没有原生双分派支持,得靠 accept() + visit() 两次虚函数调用模拟。

常见错误是把 Visitor 写成普通工具类,漏掉 visit() 的重载声明,或者在 accept() 里直接调用 visitor.visit(this) —— 这会绑定到基类指针类型,失去多态性。

  • accept() 必须是虚函数,且每个具体节点都要重写,显式调用 visitor.visit(*this)
  • visit() 在访问者中按具体节点类型重载,参数类型必须是具体类(如 const BinaryOpNode&),不能只写基类引用
  • 如果节点类型太多,Visitor 接口会膨胀;可考虑用 std::variant + std::visit 替代(C++17 起),但丢失了开闭原则的纯粹性

如何避免 const 正确性翻车

访问者常用于只读遍历(比如语义分析),但 C++ 成员函数默认非 const,容易在 visit() 里意外修改节点状态。更隐蔽的问题是:若 accept() 声明为 const,而 visit() 参数是非 const 引用,编译器会报错 —— 因为 this 是 const 指针,解引用后传给非 const 引用不合法。

  • 统一用 const T& 作为所有 visit() 参数类型(T 是具体节点类)
  • accept() 函数本身应声明为 const,除非操作确实要改节点(如优化器)
  • 如果真需要修改,把 visit() 参数改成 T&,同时确保调用方传入的是非常量对象 —— 但多数场景下,这反而暴露了设计问题

std::any 或 std::variant 能替代访问者吗

能简化简单场景,但会丢掉类型安全和编译期检查。比如用 std::variant<binaryopnode unaryopnode numbernode></binaryopnode> 存节点,再用 std::visit 分发,看起来更短:

立即学习C++免费学习笔记(深入)”;

ProcessOn
ProcessOn

免费在线流程图思维导图,专业强大的作图工具,支持多人实时在线协作

下载
std::visit([](auto& node) { /* 处理逻辑 */ }, node_variant);

但问题立刻出现:无法在访问过程中累积状态(比如符号表),也不能让不同访问者共享同一套遍历逻辑;更关键的是,一旦新增节点类型,所有 std::visit lambda 都得补分支,违反开闭原则。

  • std::variant 适合节点类型少、操作简单、且不需跨访问者复用遍历的场景
  • 访问者模式真正价值在于把「谁遍历」和「做什么」彻底分离,比如同一个 CodegenVisitor 可以被不同 AST 树复用
  • 混合使用可行:用 std::variant 表示叶子节点,访问者处理复合结构 —— 但别指望它完全取代模式本意

虚函数开销和模板化访问者的取舍

传统访问者靠虚函数实现双分派,每次 accept() + visit() 至少两次虚调用。高频遍历(如解释器执行循环)可能成瓶颈。有人用 CRTP 模板化访问者绕过虚表:

template<typename Derived> struct Visitor { void visit(const BinaryOpNode& n) { static_cast<Derived*>(this)->do_visit(n); } };

但这让访问者必须提前知道所有节点类型,破坏了「新增操作无需改节点」的初衷;而且模板实例化会让二进制体积膨胀。

  • 虚函数开销在绝大多数场景下可忽略,优先选清晰的虚函数版本
  • 只有在 profiler 明确指出 visit() 是热点,且节点类型固定不变时,才考虑模板方案
  • 别为了理论性能提前优化:访问者本身是为可维护性服务的,过度工程化反而让后续加新操作变得更麻烦

最易被忽略的一点:访问者对象的状态管理。很多人把临时变量塞进 Visitor 成员里,结果并发遍历时出问题 —— 访问者应该无状态,或明确用 thread_local 隔离,否则 debug 起来比虚函数慢十倍还难定位。

相关文章

c++速学教程(入门到精通)
c++速学教程(入门到精通)

c++怎么学习?c++怎么入门?c++在哪学?c++怎么学才快?不用担心,这里为大家提供了c++速学教程(入门到精通),有需要的小伙伴保存下载就能学习啦!

下载

本站声明:本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

WorkBuddy
WorkBuddy

腾讯云推出的AI原生桌面智能体工作台

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
java基础知识汇总
java基础知识汇总

java基础知识有Java的历史和特点、Java的开发环境、Java的基本数据类型、变量和常量、运算符和表达式、控制语句、数组和字符串等等知识点。想要知道更多关于java基础知识的朋友,请阅读本专题下面的的有关文章,欢迎大家来php中文网学习。

1570

2023.10.24

java多态详细介绍
java多态详细介绍

本专题整合了java多态相关内容,阅读专题下面的文章了解更多详细内容。

27

2025.11.27

c语言const用法
c语言const用法

const是关键字,可以用于声明常量、函数参数中的const修饰符、const修饰函数返回值、const修饰指针。详细介绍:1、声明常量,const关键字可用于声明常量,常量的值在程序运行期间不可修改,常量可以是基本数据类型,如整数、浮点数、字符等,也可是自定义的数据类型;2、函数参数中的const修饰符,const关键字可用于函数的参数中,表示该参数在函数内部不可修改等等。

565

2023.09.20

c语言const用法
c语言const用法

const是关键字,可以用于声明常量、函数参数中的const修饰符、const修饰函数返回值、const修饰指针。详细介绍:1、声明常量,const关键字可用于声明常量,常量的值在程序运行期间不可修改,常量可以是基本数据类型,如整数、浮点数、字符等,也可是自定义的数据类型;2、函数参数中的const修饰符,const关键字可用于函数的参数中,表示该参数在函数内部不可修改等等。

565

2023.09.20

lambda表达式
lambda表达式

Lambda表达式是一种匿名函数的简洁表示方式,它可以在需要函数作为参数的地方使用,并提供了一种更简洁、更灵活的编码方式,其语法为“lambda 参数列表: 表达式”,参数列表是函数的参数,可以包含一个或多个参数,用逗号分隔,表达式是函数的执行体,用于定义函数的具体操作。本专题为大家提供lambda表达式相关的文章、下载、课程内容,供大家免费下载体验。

215

2023.09.15

python lambda函数
python lambda函数

本专题整合了python lambda函数用法详解,阅读专题下面的文章了解更多详细内容。

193

2025.11.08

Python lambda详解
Python lambda详解

本专题整合了Python lambda函数相关教程,阅读下面的文章了解更多详细内容。

62

2026.01.05

硬盘接口类型介绍
硬盘接口类型介绍

硬盘接口类型有IDE、SATA、SCSI、Fibre Channel、USB、eSATA、mSATA、PCIe等等。详细介绍:1、IDE接口是一种并行接口,主要用于连接硬盘和光驱等设备,它主要有两种类型:ATA和ATAPI,IDE接口已经逐渐被SATA接口;2、SATA接口是一种串行接口,相较于IDE接口,它具有更高的传输速度、更低的功耗和更小的体积;3、SCSI接口等等。

2009

2023.10.19

c++ 字符处理
c++ 字符处理

本专题整合了c++字符处理教程、字符串处理函数相关内容,阅读专题下面的文章了解更多详细内容。

0

2026.03.17

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
C# 教程
C# 教程

共94课时 | 11.5万人学习

C 教程
C 教程

共75课时 | 5.5万人学习

C++教程
C++教程

共115课时 | 22.2万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号