如何使用Golang实现模板方法流程控制_使用Template Method控制执行步骤

模板方法模式在Go中通过接口定义流程骨架(Prepare/Execute/Cleanup)、组合结构体封装Run逻辑、函数字段注入行为实现;支持钩子扩展与条件分支,以接口约束和函数注入替代继承,复用流程并隔离变化。

模板方法(Template Method)是一种行为型设计模式,它定义一个操作中算法的骨架,将一些步骤延迟到子类中实现。Golang 没有继承机制,但可以通过组合 + 接口 + 函数字段灵活模拟该模式,实现清晰、可扩展的流程控制。

用接口定义流程契约

先定义一个接口,声明“模板方法”要执行的各阶段,比如 PrepareExecuteCleanup。每个方法不带参数或只接收上下文,保持协议轻量:

type Workflow interface {
    Prepare() error
    Execute() error
    Cleanup() error
}

这个接口就是流程的“骨架签名”,不关心具体怎么实现,只约定必须提供哪些环节。

用结构体封装模板逻辑

定义一个通用控制器结构体,内嵌 Workflow 接口,并实现统一的 Run 方法——这就是真正的“模板方法”:

type Controller struct {
    w Workflow
}

func (c *Controller) Run() error {
    if err := c.w.Prepare(); err != nil {
        return err
    }
    if err := c.w.Execute(); err != nil {
        return err
    }
    return c.w.Cleanup()
}

所有流程都走同一路径:准备 → 执行 → 清理。错误会短路中断,保证步骤可控。

用匿名结构或函数字段定制行为

Golang 中不必写多个 struct 子类。你可以直接构造一个满足 Workflow 接口的匿名值,或用字段注入函数:

  • 方式一:匿名结构实现
workflow := &struct{ Workflow }{
    Workflow: &MyTask{
        Data: "hello",
    },
}
  • 方式二:函数字段更轻量(推荐)
task := &FuncWorkflow{
    PrepareFn: func() error { log.Println("preparing..."); return nil },
    ExecuteFn: func() error { log.Println("running..."); return nil },
    CleanupFn: func() error { log.Println("cleaning..."); return nil },
}
controller := &Controller{w: task}
controller.Run()

只需实现三个函数,无需定义新类型,适合一次性流程或配置化任务。

支持钩子与条件分支

真实流程常需前置校验、失败重试或跳过某步。可在模板方法中预留钩子,例如:

  • 添加 BeforeRunAfterRun 回调字段
  • Execute 前检查 ShouldExecute 返回 bool
  • Cleanup 改为 Cleanup(err error),根据错误类型决定清理策略

这样既保持主干稳定,又让扩展点显式、安全、无侵入。

基本上就这些。Golang 的模板方法不是靠继承“强制子类实现”,而是靠接口约束 + 组合封装 + 函数注入来达成同样的目的:复用流程骨架,隔离变化细节。不复杂但容易忽略的是——别把所有逻辑塞进 Run,要把“可变”和“不变”真正切分开。