Go中实现访问者模式核心是用接口解耦数据结构与操作,通过元素的Accept方法显式调用访问者对应Visit方法,实现类型安全的伪双分派,符合开闭原则且避免反射开销。

在 Go 语言中实现访问者模式,核心在于用接口解耦数据结构与操作逻辑,避免为每种新操作修改已有类型。Go 没有传统面向对象的“方法重载”或“双分派”,但可通过组合、接口和类型断言巧妙模拟访问者行为。
定义元素接口与访问者接口
先明确谁被访问、谁去访问。通常让数据结构(如 AST 节点、配置项、树形对象)实现 Accept(Visitor) 方法,而访问者实现一组 VisitXXX() 方法:
-
元素接口:声明
Accept(v Visitor),不关心具体访问逻辑 -
访问者接口:按需定义
VisitStringLit(*StringLit)、VisitNumberLit(*NumberLit)等,每个方法接收具体类型指针
用类型断言 + 接口实现“伪双分派”
Go 不支持运行时自动匹配访问方法,所以要在 Accept 中显式调用访问者对应方法:
func (s *StringLit) Accept(v Visitor) {
v.VisitStringLit(s) // 直接调用,不靠反射
}
这样既保持类型安全,又避免反射开销。新增一种节点类型?只需实现 Accept 并在访问者接口中添加对应方法 —— 符合开闭原则。
立即学习“go语言免费学习笔记(深入)”;
避免循环依赖:用内部包或函数式访问者
当元素和访问者定义在不同包时,容易出现 import 循环。可行解法有:
- 把
Visitor接口放在元素所在包(如ast包定义Visitor),访问者实现放外部包 - 改用函数式风格:定义
type VisitFunc func(interface{}),配合switch v := node.(type)分支处理(适合简单场景) - 用空接口 + 类型检查做轻量访问者,牺牲部分类型安全换灵活性
实战建议:从遍历需求出发,别为模式而模式
访问者模式真正价值在于统一处理异构结构(如表达式树、配置嵌套、协议消息)。常见适用场景:
- 语法树遍历:生成代码、校验语义、计算复杂度
- 序列化/反序列化策略切换(JSON/YAML/Proto)
- 权限检查:对不同资源类型执行差异化鉴权逻辑
若只有 2–3 种操作且结构稳定,直接用 switch 或方法字段更简洁。访问者适合操作多、结构稳、未来易扩展的场景。
基本上就这些。Go 的访问者不是照搬 Java 写法,而是用接口组合 + 显式分发来达成同样目标 —— 清晰、可控、无反射负担。










