Go通过接口与指针接收者实现多态:接口定义行为契约,指针接收者支持状态修改与动态分发,运行时根据接口的(type, value)对调用对应类型方法。

在 Go 语言中,没有传统面向对象语言(如 Java、C++)中的“类继承”和“虚函数表”,但可以通过 接口 + 指针接收者 的组合,实现类似多态的动态行为调用——即同一接口变量,在运行时根据实际指向的类型,调用对应的具体方法。
接口定义统一行为契约
接口是 Go 实现多态的核心。它不关心具体类型,只约定“能做什么”。例如:
type Shape interface {
Area() float64
Name() string
}
只要某个类型实现了 Area() 和 Name() 方法,它就自动满足 Shape 接口,无需显式声明“implements”。
立即学习“go语言免费学习笔记(深入)”;
指针接收者决定方法是否可修改状态
使用指针接收者(func (s *Square) Area() float64)而非值接收者,有两个关键作用:
- 避免大结构体复制,提升性能
- 允许方法内部修改接收者字段(如缓存计算结果、更新计数器等)
- 更重要的是:当接口变量存储的是指针类型时,调用的正是指针接收者方法;若存储值类型,则只能调用值接收者方法 —— 这直接影响多态行为的一致性
示例中,Square 和 Circle 都用指针接收者实现 Shape,这样 *Square{} 和 *Circle{} 都能赋给 Shape 变量,且各自调用自己版本的 Area()。
运行时动态分发:接口底层是 (type, value) 对
Go 接口变量在内存中包含两个字:类型信息(type)和数据指针(value)。当调用 shape.Area() 时,运行时根据 type 查找对应类型的 Area 方法地址,再跳转执行 —— 这就是动态行为调用的本质。
代码示例:
var s Shape
s = &Square{Side: 5}
fmt.Println(s.Name(), s.Area()) // "Square 25"
s = &Circle{Radius: 3}
fmt.Println(s.Name(), s.Area()) // "Circle 28.274..."
同一变量 s,两次赋值不同指针类型,调用的却是各自独立实现的方法,行为完全动态。
注意值 vs 指针接收者的隐式转换陷阱
如果某类型只用值接收者实现了接口,那它的值和指针都能满足该接口;但如果只用指针接收者实现,则只有指针能赋给接口变量,值类型会编译报错。
常见错误写法:
s := Square{Side: 5}
var sh Shape = s // ✅ 编译通过(假设 Square 有值接收者方法)
sh = &s // ✅ 也行
但如果 Square 只有 *Square 的方法:
var sh Shape = s // ❌ cannot use s (type Square) as type Shape in assignment
因此,若希望支持多态调用并允许修改状态,应统一使用指针接收者,并始终以 &T{} 方式构造实例后传入接口。










