您好,欢迎访问宜昌市隼壹珍商贸有限公司
400 890 5375模板方法模式在Go中需用接口+结构体组合模拟,通过定义Processor接口和Workflow执行器实现流程控制,钩子方法参数应统一为共享state指针,避免嵌入具体类型导致方法集不匹配。
Go 没有抽象类、不能定义“仅声明不实现”的方法,所以 Template Method Pattern 必须靠接口 + 结构体嵌入 + 显式调用约定来实现。核心思路是:定义一个公共执行流程(如 Execute()),把可变步骤抽成接口方法,由具体类型实现。
典型结构是声明一个 Processor 接口,包含 Setup()、DoWork()、TearDown() 等钩子;再写一个通用执行器结构体,持有该接口并按序调用——这就是“模板”。实际行为由传入的具体实现决定。
type Processor interface {
Setup()
DoWork()
TearDown()
}
type Workflow struct {
p Processor
}
func (w *Wor
kflow) Execute() {
w.p.Setup()
w.p.DoWork()
w.p.TearDown()
}
// 具体实现
type FileProcessor struct{}
func (f FileProcessor) Setup() { fmt.Println("open file") }
func (f FileProcessor) DoWork() { fmt.Println("parse content") }
func (f FileProcessor) TearDown() { fmt.Println("close file") }
// 使用
wf := &Workflow{p: FileProcessor{}}
wf.Execute()
如果不同实现需要不同上下文(比如有的要传 *os.File,有的要传 context.Context),就不能直接用空接口或硬编码参数。常见做法是:
state 结构体指针,各实现按需读写字段Workflow 字段中,在 Execute() 前初始化例如,加一个 State 字段后,Setup() 可以初始化资源,DoWork() 直接用 w.state.Input,不用改签名。
如果用匿名字段嵌入具体类型(如 struct{ FileProcessor })来“复用”钩子实现,要注意 Go 的方法集规则:只有**值类型嵌入**时,外部结构体才拥有其指针方法;而**指针嵌入**会导致方法集不被提升。更稳妥的方式是显式组合接口,而非嵌入具体类型。
容易踩的坑包括:
struct{ FileProcessor } 能自动满足 Processor 接口——它不能,因为 FileProcessor 是值类型,其方法集只包含值接收者方法;若你定义的是指针接收者方法(如 func (f *FileProcessor) Setup()),那必须嵌入 *FileProcessor
Execute() 内部调用钩子时,传了错误的 receiver(比如用了值而非指针),导致状态未更新最简、最可控的做法始终是:用接口字段 + 显式赋值,不依赖嵌入自动提升。