Go通过接口+方法组合实现访问者模式:Element接口定义Accept方法,各具体类型实现并调用visitor对应Visit方法;Visitor接口声明VisitBook、VisitVideo等,新增操作只需实现新访问者,无需修改原有结构。

用 Go 实现访问者模式(Visitor Pattern),核心是把“数据结构”和“操作逻辑”拆开,让新增操作不用改已有类型,提升可维护性。Go 没有传统面向对象的继承与虚函数,但可以通过接口 + 方法组合 + 类型断言来自然落地这个模式。
定义被访问的数据结构(Element 接口)
所有要被访问的类型都实现一个统一接口,比如 Accept(visitor Visitor) 方法,把自身传给访问者:
- 每个具体结构(如
Book、Video)实现Accept,内部调用visitor.VisitXXX(this) - 避免在结构体里硬编码访问者逻辑,只负责“转发”
定义访问者接口(Visitor 接口)
声明一组 VisitXXX 方法,对应每种被访问类型:
- 例如
VisitBook(*Book)、VisitVideo(*Video) - 访问者实现该接口,就能为不同结构提供定制行为
- 新增一种操作?只需新增一个访问者实现,不碰原有结构代码
用类型断言或反射处理“双分派”(Go 的替代方案)
经典 Visitor 依赖运行时多态(双分派),Go 用接口+方法调用模拟:在 Accept 中显式调用访问者对应方法,靠编译期类型确定目标函数:
立即学习“go语言免费学习笔记(深入)”;
func (b *Book) Accept(v Visitor) { v.VisitBook(b) }- 访问者内部可自由读取字段、调用方法,甚至修改(如果传的是指针)
- 不需要反射,清晰安全;也不需要泛型约束(Go 1.18+ 可选,但非必需)
实际使用示例:内容计费与导出
假设你有一组媒体资源 Book 和 Video,想分别支持“计算价格”和“生成 JSON 描述”:
- 写一个
PricingVisitor实现VisitBook/VisitVideo,各自按规则算价 - 再写一个
JSONExportVisitor实现同样两个方法,输出结构化数据 - 遍历资源列表,对每个元素调
.Accept(pricingV)或.Accept(exportV)即可切换行为
基本上就这些。Go 的 Visitor 不追求语法糖,重在职责清晰——结构管“是什么”,访问者管“怎么用”。不复杂但容易忽略的是:Accept 方法必须由每个具体类型自己实现,不能靠嵌入共用,否则无法精准路由到对应 Visit 方法。










