模板方法模式在go语言中通过接口与结构体嵌套实现,其核心在于定义算法骨架并延迟步骤实现。具体步骤为:1. 定义接口声明各步骤;2. 创建基础结构体包含模板方法run();3. 具体任务结构体嵌入基础结构体并实现特定步骤;4. 可选步骤可在基类提供默认实现;5. 通过接口组合提升流程扩展性。

在 Go 语言中,模板方法模式的实现不像 Java 或 C++ 那样依赖继承和虚函数机制。Go 没有“类”的概念,也没有继承语法,但可以通过接口(interface)与结构体嵌套(embedding)来模拟类似行为。其中,通过嵌入结构体扩展行为是一种非常实用的方式,可以很好地实现模板方法模式。

什么是模板方法模式?
模板方法模式的核心思想是:在一个方法中定义算法的骨架,而将一些步骤延迟到子类中去实现。这样可以在不改变算法结构的前提下,允许子类重新定义某些步骤。

在 Go 中,我们通常用一个接口定义算法流程中的各个可变步骤,然后通过结构体嵌套的方式提供默认实现或抽象逻辑。
立即学习“go语言免费学习笔记(深入)”;
使用嵌入结构体实现模板方法
假设我们要实现一个任务执行流程:准备资源 -> 执行任务 -> 清理资源。不同任务的具体准备和清理方式可能不同,但整体流程保持一致。

定义接口与基础结构体
首先定义一个接口:
type Task interface {
Prepare()
Execute()
Cleanup()
}然后创建一个包含模板方法的基础结构体:
type BaseTask struct{}
func (b *BaseTask) Run() {
b.Prepare()
b.Execute()
b.Cleanup()
}注意:这里
BaseTask并没有实现
Prepare()、
Execute()和
Cleanup()方法,因此它不能直接作为
Task接口使用。
嵌入结构体并重写行为
接下来,我们可以定义具体的任务类型,并嵌入
BaseTask,同时实现各自的行为:
type DownloadTask struct {
BaseTask
}
func (d *DownloadTask) Prepare() {
fmt.Println("Preparing network connection...")
}
func (d *DownloadTask) Execute() {
fmt.Println("Downloading file...")
}
func (d *DownloadTask) Cleanup() {
fmt.Println("Closing connection...")
}调用时:
task := &DownloadTask{}
task.Run()输出:
Preparing network connection... Downloading file... Closing connection...
这种方式的关键在于:
- 利用了结构体嵌套来“继承”公共方法(如
Run
) - 子结构体只需实现特定方法即可定制行为
- 整体流程由父结构体控制,符合模板方法的设计理念
如何处理可选步骤或默认行为?
有时某些步骤不是必须的,或者希望提供默认实现。这时可以考虑在基类中提供空实现:
func (b *BaseTask) Prepare() {
// 默认不做任何事
}
func (b *BaseTask) Cleanup() {
// 默认不做任何事
}这样,子类可以选择性地重写这些方法。比如有的任务不需要清理操作,就不需要实现
Cleanup()。
小技巧:通过接口组合更灵活地定义流程
如果你希望流程本身也更具扩展性,可以把流程方法提取为接口的一部分:
type Task interface {
Prepare()
Execute()
Cleanup()
Run()
}然后让每个具体任务决定是否自定义整个流程,或者复用默认的
Run()实现。
总结一下
- Go 中没有传统意义上的继承,但结构体嵌入提供了类似的能力
- 模板方法可以通过定义统一的运行流程(如
Run()
),再配合接口和方法覆盖实现 - 基类中可以提供默认实现,子类选择性重写
- 可以根据实际需求决定是否暴露完整流程接口
基本上就这些。核心思路是利用结构体嵌套+接口约束,来达到“控制流程、扩展行为”的效果。










