Golang模板方法模式如何实现_Golang模板方法模式解析
发布时间 - 2026-02-01 00:00:00 点击率:次Go中模板方法模式用函数字段或接口实现:函数字段轻量适合无状态流程,接口组合适合需共享状态的场景;必填步骤须显式校验非空,禁用嵌入模拟继承。
Go 里没有抽象类和继承,所以不能照搬 Java 的 abstract class + final method 写法;但模板方法模式的本质不是语法,而是「流程骨架固定、步骤实现可插拔」——这完全可以用函数字段或接口组合干净落地。
用结构体 + 函数字段实现最轻量模板
适合一次性流程、CLI 命令、HTTP 中间件钩子等场景。不引入接口,零类型定义,测试时直接传 mock 函数即可。
关键点:
-
Run()是唯一入口,流程顺序硬编码,不可覆盖 - 必填步骤(如
Validate、Process)在Run()中显式判空并返回错误,不默认 fallback - 可选步骤(如
Teardown)允许为nil,调用前必须判空,否则 panic - 闭包可捕获上下文变量,比接口更灵活,但无法复用步骤逻辑
type PaymentProcessor struct {
Validate func() error
Charge func() error
Notify func() error
}
func (p *PaymentProcessor) Execute() error {
if p.Validate == nil {
return fmt.Errorf("Validate not set")
}
if err := p.Validate(); err != nil {
return err
}
if p.Charge == nil {
return fmt.Errorf("Charge not set")
}
if err := p.Charge(); err != nil {
return err
}
if p.Notify != nil {
_ = p.Notify()
}
return nil
}
用接口 + 显式实现替代“抽象基类”
当多个流程共享状态(如 DB 连接、日志器、重试配置),或步骤间有依赖(如 Transform 需要 Validate 的输出),接口组合更清晰、易维护。
注意陷阱:
- 别把所有步骤塞进一个大接口(如
Processor含 8 个方法),按业务切分小接口(如Validator、Persister)更利于组合和 mock - 结构体实现接口时,可内嵌通用字段(如
*sql.DB),避免每个方法重复传参 - Go 不支持方法重写:若用匿名嵌入“默认实现”,
p.Transform()调用的仍是嵌入字段的方法,不会自动路由
到你新写的同名方法
为什么别用 embed + 匿名字段模拟继承
常见误操作:定义一个 BaseProcessor 结构体,含默认 Setup() 和 Teardown(),再让 CoffeeProcessor 嵌入它并重写 Brew()。结果是:CoffeeProcessor.Brew() 永远调不到你重写的版本,因为 Go 的方法调用目标由值的静态类型决定,不是运行时动态绑定。
真实行为:
-
type CoffeeProcessor struct{ BaseProcessor }→CoffeeProcessor.Brew()调用的是BaseProcessor.Brew() - 要真正替换,必须显式重定向:
func (c *CoffeeProcessor) Brew() { ... },且不能依赖嵌入字段自动转发 - 这种写法徒增嵌套层级,语义模糊,违背 Go 的“显式优于隐式”原则
接口组合 vs 函数字段:怎么选
判断依据就一条:是否需要跨步骤共享状态或复用逻辑。
- 纯流程控制(如日志上报三步:校验→序列化→发送),用函数参数最直接:
Process(data, json.Marshal, http.Post) - 需共享缓存、重试次数、context 或中间结果(如
Transform()输出要传给Persist()),必须用接口+结构体,靠字段承载状态 - 混合场景(如大部分步骤固定,仅一两个可插拔),可将可变步骤声明为接口字段,其余逻辑写死在
Run()里
最容易被忽略的一点:无论哪种方式,**必填步骤绝不能在 NewXXX() 构造函数里偷偷赋默认实现**——这会让使用者误以为流程已完备,实际掩盖了契约缺失,破坏可验证性。
# java
# js
# json
# go
# golang
# 编码
# 路由
# 为什么
# sql
# 中间件
# 构造函数
# 结构体
# 继承
# 接口
# class
# Struct
# 闭包
# nil
# transform
# http
# 重写
# 必填
# 到你
# 重试
# 复用
# 的是
# 插拔
# 切分
# 多个
# 可以用
相关栏目:
【
网站优化151355 】
【
网络推广146373 】
【
网络技术251813 】
【
AI营销90571 】
相关推荐:
如何制作新型网站程序文件,新型止水鱼鳞网要拆除吗?
Laravel如何实现图片防盗链功能_Laravel中间件验证Referer来源请求【方案】
如何在IIS7上新建站点并设置安全权限?
哪家制作企业网站好,开办像阿里巴巴那样的网络公司和网站要怎么做?
Laravel项目怎么部署到Linux_Laravel Nginx配置详解
如何在阿里云虚拟服务器快速搭建网站?
微信h5制作网站有哪些,免费微信H5页面制作工具?
DeepSeek是免费使用的吗 DeepSeek收费模式与Pro版本功能详解
Laravel如何使用Eloquent ORM进行数据库操作?(CRUD示例)
JavaScript中如何操作剪贴板_ClipboardAPI怎么用
Laravel如何实现用户密码重置功能?(完整流程代码)
制作无缝贴图网站有哪些,3dmax无缝贴图怎么调?
历史网站制作软件,华为如何找回被删除的网站?
如何用AWS免费套餐快速搭建高效网站?
Laravel怎么做数据加密_Laravel内置Crypt门面的加密与解密功能
Midjourney怎样加参数调细节_Midjourney参数调整技巧【指南】
Laravel模型关联查询教程_Laravel Eloquent一对多关联写法
Laravel如何使用.env文件管理环境变量?(最佳实践)
LinuxShell函数封装方法_脚本复用设计思路【教程】
Laravel如何实现API资源集合?(Resource Collection教程)
JavaScript实现Fly Bird小游戏
Laravel怎么实现前端Toast弹窗提示_Laravel Session闪存数据Flash传递给前端【方法】
悟空识字怎么关闭自动续费_悟空识字取消会员自动扣费步骤
大连 网站制作,大连天途有线官网?
Laravel如何清理系统缓存命令_Laravel清除路由配置及视图缓存的方法【总结】
如何用低价快速搭建高质量网站?
软银砸40亿美元收购DigitalBridge 强化AI资料中心布局
如何实现javascript表单验证_正则表达式有哪些实用技巧
Thinkphp 中 distinct 的用法解析
打造顶配客厅影院,这份100寸电视推荐名单请查收
如何用PHP工具快速搭建高效网站?
厦门模型网站设计制作公司,厦门航空飞机模型掉色怎么办?
Laravel如何使用withoutEvents方法临时禁用模型事件
北京的网站制作公司有哪些,哪个视频网站最好?
如何在万网自助建站平台快速创建网站?
Java遍历集合的三种方式
laravel怎么配置Redis作为缓存驱动_laravel Redis缓存配置教程
EditPlus中的正则表达式 实战(4)
百度浏览器如何管理插件 百度浏览器插件管理方法
网站设计制作书签怎么做,怎样将网页添加到书签/主页书签/桌面?
HTML5空格和margin有啥区别_空格与外边距的使用场景【说明】
实例解析Array和String方法
Laravel如何实现文件上传和存储?(本地与S3配置)
如何在自有机房高效搭建专业网站?
Laravel如何实现用户注册和登录?(Auth脚手架指南)
在线制作视频的网站有哪些,电脑如何制作视频短片?
Laravel安装步骤详细教程_Laravel环境搭建指南
Windows家庭版如何开启组策略(gpedit.msc)?(安装方法)
bootstrap日历插件datetimepicker使用方法
如何快速搭建高效WAP手机网站吸引移动用户?


