模板方法在Go中通过函数参数或接口注入可变步骤,核心是固定流程骨架与分离变化点;函数参数适合简单场景,接口+结构体嵌入适用于多步骤、需共享状态的场景,且必须显式实现接口方法。
Go 没有继承和抽象类,所以不能像 Java 那样靠 abstract func() 强制子类实现。但模板方法的本质不是语法机制,而是「把不变的流程提出来,把可变的步骤留给调用方决定」。关键在于:用函数参数或接口把步骤“注入”到主流程中。
适合步骤少、逻辑简单、不需复用步骤实现的场景。比如日志上报前的预处理流程:校验 → 序列化 → 发送 → 清理,其中只有序列化和发送可能变化。
Process 函数接收 serialize 和 send 两个 func([]byte) error 类型参数,流程顺序完全固定json.Marshal 还是
protobuf.Marshal,不影响主干逻辑func Process(data interface{}, serialize func(interface{}) ([]byte, error), send func([]byte) error) error {
if data == nil {
return errors.New("data is nil")
}
b, err := serialize(data)
if err != nil {
return err
}
return send(b)
}
当步骤多、需要共享状态(如配置、缓存、上下文)、或步骤间有依赖时,接口更清晰。重点不是“继承”,而是让调用方实现接口,再把实例传给模板函数。
Processor 接口,含 Validate、Transform、Persist 等方法Run 按序调用这些方法,不关心具体实现*sql.DB 或 context.Context),避免每个方法重复传参type Processor interface {
Validate() error
Transform() error
Persist() error
}
func Run(p Processor) error {
if err := p.Validate(); err != nil {
return err
}
if err := p.Transform(); err != nil {
return err
}
return p.Persist()
}
有人尝试用嵌入一个“默认实现”结构体,再覆盖部分方法,看似接近 OOP 的模板方法。但 Go 不支持方法重写语义,匿名字段只是字段提升,不会改变调用目标——p.Transform() 调用的仍是嵌入字段的方法,而非你新定义的同名方法,除非显式重定向。
A 嵌入 DefaultProcessor,又定义了 Transform() 方法,A.Transform() 是新的实现,但 A 本身仍满足 Processor 接口,因为它的方法集包含 Transform
Transform 依赖 Validate 设置的某个字段,但接口没声明这个契约。这种时候,文档比代码更重要。