答案:在Golang中通过接口实现访问者模式,分离数据结构与操作,定义Shape接口和Visitor接口,让Circle和Rectangle实现Accept方法,分别调用对应访问者,从而支持扩展面积计算、信息打印等行为而不修改原有结构。

在Golang中实现访问者模式,核心是将数据结构与作用于其上的操作分离。这种设计让你可以在不修改原有结构的前提下,为不同类型的数据添加新的行为。特别适合处理复杂、多类型的树形或组合结构,比如AST(抽象语法树)、配置解析等场景。
定义访问者接口和可访问对象
访问者模式包含两个关键角色:访问者(Visitor)和被访问元素(Element)。每个元素实现一个 Accept 方法,接收访问者作为参数。
注意: Go 没有继承,因此通过接口模拟多态行为。假设我们有一组形状结构(如圆形、矩形),想分别计算面积和打印信息,但不想把这些逻辑耦合进结构体本身。
定义接口:
立即学习“go语言免费学习笔记(深入)”;
type Shape interface {
Accept(v Visitor)
}
type Visitor interface {
VisitCircle(c Circle)
VisitRectangle(r Rectangle)
}
每个具体形状实现 Accept 方法,并调用访问者对应的方法。
实现具体的数据结构
以 Circle 和 Rectangle 为例:
type Circle struct {
Radius float64
}
func (c *Circle) Accept(v Visitor) {
v.VisitCircle(c)
}
type Rectangle struct {
Width, Height float64
}
func (r *Rectangle) Accept(v Visitor) {
v.VisitRectangle(r)
}
这样,结构体把“如何处理自己”的决定权交给访问者。
定义不同类型的访问者
你可以创建多个访问者来执行不同任务。例如:
- 面积计算器
- 信息打印器
计算面积的访问者:
type AreaCalculator struct {
Area float64
}
func (a AreaCalculator) VisitCircle(c Circle) {
a.Area += 3.14159 c.Radius c.Radius
}
func (a AreaCalculator) VisitRectangle(r Rectangle) {
a.Area += r.Width * r.Height
}
打印信息的访问者:
type InfoPrinter struct{}
func (i InfoPrinter) VisitCircle(c Circle) {
println("Circle: radius =", c.Radius)
}
func (i InfoPrinter) VisitRectangle(r Rectangle) {
println("Rectangle: width =", r.Width, "height =", r.Height)
}
使用访问者遍历结构
当你有一组形状时,统一调用它们的 Accept 方法即可触发相应行为:
shapes := []Shape{
&Circle{Radius: 3},
&Rectangle{Width: 4, Height: 5},
&Circle{Radius: 2},
}
// 计算总面积
calculator := &AreaCalculator{}
for _, s := range shapes {
s.Accept(calculator)
}
println("Total area:", calculator.Area)
// 打印信息
printer := &InfoPrinter{}
for _, s := range shapes {
s.Accept(printer)
}
新增操作(如序列化、校验)只需添加新访问者,无需改动现有形状代码,符合开闭原则。
基本上就这些。Go 虽无虚函数表,但通过接口和方法绑定也能优雅实现访问者模式,尤其适合需要频繁扩展操作的稳定数据结构。关键是让每个类型明确知道自己能被谁访问。










