Go中实现模板方法模式的核心是接口定义可变步骤、结构体封装固定流程,通过组合与嵌入实现稳定骨架与可插拔行为;例如ReportRunner.Run()调用load→g.Format()→g.Export(),具体类型如JSONReport/HTMLReport仅需实现接口方法。

在 Go 语言中实现模板方法模式,核心思路是:用接口定义算法骨架的“钩子”,用结构体组合封装不变逻辑,把可变步骤延迟到具体类型中实现。Go 没有继承和抽象方法,但通过组合 + 接口 + 嵌入,完全可以表达模板方法的本质——稳定流程控制 + 可插拔行为。
用接口声明“可变步骤”
模板方法的关键是把算法中可能变化的部分抽成接口方法。例如一个报告生成流程:加载数据 → 格式化 → 输出。其中“格式化”和“输出”因报告类型而异,就定义为接口:
- 定义 ReportGenerator 接口,含 Format() 和 Export() 方法
- 不暴露 LoadData —— 它是固定步骤,由模板方法内部调用
用结构体封装“算法骨架”
创建一个通用执行器(如 ReportRunner),它持有 ReportGenerator 接口,并实现完整流程:
- ReportRunner.Run() 是模板方法:依次调用 load()、g.Format()、g.Export()
- load() 是私有函数或内联逻辑,对所有子类一致
- Run 不关心 g 具体是谁,只依赖接口契约
让具体类型实现接口并注入
不同报告类型只需实现接口,无需继承。例如 JSONReport 和 HTMLReport:
立即学习“go语言免费学习笔记(深入)”;
- JSONReport 实现 Format() 返回 JSON 字节,Export() 写入文件
- HTMLReport 实现 Format() 返回 HTML 字符串,Export() 启动 HTTP 服务
- 使用时 new(ReportRunner).Run(&JSONReport{}) 即可触发完整流程
支持钩子与默认行为(可选增强)
若某些步骤可选或有默认实现,可在接口中提供空实现或用函数字段:
- 在 ReportGenerator 接口中增加 BeforeExport() func(),默认为 nil
- Run 中判断 if g.BeforeExport != nil { g.BeforeExport() }
- 调用方可选择性设置钩子,不影响主流程
不复杂但容易忽略:Go 的模板方法不靠语法强制,而靠设计约定和接口清晰度来保障骨架稳定性。只要接口职责单一、Run 方法不暴露可重写点,就达成了模式意图。










