访问者模式是一种行为型设计模式,它将数据结构与作用于其上的操作分离,使操作可独立变化;核心是在不修改原有类的前提下,为稳定结构动态添加新功能,典型用于AST遍历、文档渲染等多操作场景。
访问者模式是一种行为型设计模式,它把数据结构和作用于结构上的操作分离,使操作可以独立变化。核心思想是:不修改原有类的前提下,为复杂对象结构(比如树、图、组合体)动态添加新功能。
典型场景是“一个结构稳定但操作多变”——比如编译器遍历抽象语法树(AST)、文档渲染引擎处理不同节点、序列化/校验/统计等对同一对象树的多种遍历需求。这时,若每个节点类都硬编码所有操作,会违反开闭原则;而访问者模式让操作集中到访客类中,新增功能只需加新访客,不碰节点代码。
访问者模式包含四个核心角色:
visit(ExpressionNode&)、visit(IfNode&)),支持重载区分不同节点CodeGenerator、Validator、SizeCounter
lement(元素):声明 accept(Visitor&) 方法,负责将自身传给访问者accept(Visitor&) 或 traverse(Visitor&)),内部调用各子元素的 accept()
C++ 原生只支持单分派(靠虚函数按运行时 this 类型分发),而访问者需要“根据访问者类型 + 元素类型”双重决定调用哪个 visit 函数——这靠 两次虚函数调用 实现:
visitor.visit(*this) —— 第一次分派:确定 visitor 类型visit() 重载函数接收具体元素引用 —— 第二次分派:编译期靠参数类型绑定到对应实现注意:元素类中的 accept() 必须是虚函数,且 visitor 参数常以非 const 引用或指针传递(便于后续重载);visit 方法签名需严格匹配元素类型,否则无法触发正确重载。
面对深层嵌套、异构节点、循环引用或需上下文状态的结构,可这样增强访问者:
accept(),控制遍历顺序(前序/后序/跳过某分支)const Element& 参数;对修改结构场景(如优化 AST),用 std::shared_ptr 并返回新节点访问者模式不是银弹,使用时要注意:
IRBuilderVisitor 和 VerifierVisitor 分离)