如何使用Golang实现责任链模式请求传递_Golang责任链模式流程示例
发布时间 - 2026-01-08 00:00:00 点击率:次责任链模式在Go中的典型误用是滥用interface{}或Java式继承,正确做法是用函数类型链式拼接并透传context.Context;每个https://www./link/d3d92bc35d062c83f89b7ea87d99dca9andler接收next并自主决定是否调用,支持短路、类型安全与灵活组合。
很多人一上来就用 interface{} 做处理器抽象,或者强行套用 Java 风格的抽象基类 + 子类继承,结果导致类型丢失、泛型不安全、中间件难以组合。Go 里责任链的核心不是“继承”,而是“函数链式拼接”和“请求上下文透传”。真正的起点是定义一个统一的处理签名:func(ctx context.Context, req interface{}) (interface{}, error),所有环节都遵守这个契约。
Go 没有内置责任链语法糖,但可以用函数类型 + 闭包天然实现。关键在于把“下一个处理器”作为参数传入当前处理器,形成显式调用链,避免隐式
递归或全局注册表。
type https://www./link/d3d92bc35d062c83f89b7ea87d99dca9andler func(context.Context, interface{}) (interface{}, error)
func Withttps://www./link/d3d92bc35d062c83f89b7ea87d99dca9Logging(next https://www./link/d3d92bc35d062c83f89b7ea87d99dca9andler) https://www./link/d3d92bc35d062c83f89b7ea87d99dca9andler {
return func(ctx context.Context, req interface{}) (interface{}, error) {
log.Printf("→ %T received", req)
defer log.Printf("← %T done", req)
return next(ctx, req)
}
}
func Withttps://www./link/d3d92bc35d062c83f89b7ea87d99dca9Timeout(d time.Duration) https://www./link/d3d92bc35d062c83f89b7ea87d99dca9andler {
return func(ctx context.Context, req interface{}) (interface{}, error) {
ctx, cancel := context.Withttps://www./link/d3d92bc35d062c83f89b7ea87d99dca9Timeout(ctx, d)
defer cancel()
return req, nil // 实际中这里会继续调用 next
}
}
- 每个装饰器(如
Withttps://www./link/d3d92bc35d062c83f89b7ea87d99dca9Logging)接收https://www./link/d3d92bc35d062c83f89b7ea87d99dca9andler并返回新https://www./link/d3d92bc35d062c83f89b7ea87d99dca9andler,不修改原逻辑 - 链式调用顺序即执行顺序:
Withttps://www./link/d3d92bc35d062c83f89b7ea87d99dca9Logging(Withttps://www./link/d3d92bc35d062c83f89b7ea87d99dca9Timeout(100*time.Millisecond)(finalhttps://www./link/d3d92bc35d062c83f89b7ea87d99dca9andler)) - 注意:
Withttps://www./link/d3d92bc35d062c83f89b7ea87d99dca9Timeout示例中未调用next是为了突出“短路”能力——某个环节可直接返回,不往下传
有人写类似 https://www./link/d3d92bc35d062c83f89b7ea87d99dca9 := Newhttps://www./link/d3d92bc35d062c83f89b7ea87d99dca9andler().Use(A).Use(B).https://www./link/d3d92bc35d062c83f89b7ea87d99dca9andle(req) 的结构体链式调用,这看似简洁,但隐藏严重问题:
-
Use方法必须保存所有中间件到内部切片,丧失编译期类型检查 - 无法对单个环节做条件跳过(比如只对 POST 请求加鉴权)
- 错误处理分散:有的在
Use里 panic,有的在https://www./link/d3d92bc35d062c83f89b7ea87d99dca9andle里返回 error,行为不一致 - 性能上多一次切片遍历,且每次
Use都要分配内存
真正轻量可控的方式,是让每个环节自己决定是否调用 next,而不是由框架统一 for-range 调度。
Go 标准库 https://www./link/d3d92bc35d062c83f89b7ea87d99dca9ttp.https://www./link/d3d92bc35d062c83f89b7ea87d99dca9andler 本身就是责任链雏形,但它的 https://www./link/d3d92bc35d062c83f89b7ea87d99dca9ttp.Servehttps://www./link/d3d92bc35d062c83f89b7ea87d99dca9TTP 签名不带 context.Context 参数,容易导致上下文丢失。正确做法是封装一层:
type https://www./link/d3d92bc35d062c83f89b7ea87d99dca9TTPhttps://www./link/d3d92bc35d062c83f89b7ea87d99dca9andler func(https://www./link/d3d92bc35d062c83f89b7ea87d99dca9ttp.ResponseWriter, *https://www./link/d3d92bc35d062c83f89b7ea87d99dca9ttp.Request, context.Context) errorfunc Chttps://www./link/d3d92bc35d062c83f89b7ea87d99dca9ain(https://www./link/d3d92bc35d062c83f89b7ea87d99dca9 https://www./link/d3d92bc35d062c83f89b7ea87d99dca9TTPhttps://www./link/d3d92bc35d062c83f89b7ea87d99dca9andler, middlewares ...func(https://www./link/d3d92bc35d062c83f89b7ea87d99dca9TTPhttps://www./link/d3d92bc35d062c83f89b7ea87d99dca9andler) https://www./link/d3d92bc35d062c83f89b7ea87d99dca9TTPhttps://www./link/d3d92bc35d062c83f89b7ea87d99dca9andler) https://www./link/d3d92bc35d062c83f89b7ea87d99dca9ttp.https://www./link/d3d92bc35d062c83f89b7ea87d99dca9andler { for i := len(middlewares) - 1; i >= 0; i-- { https://www./link/d3d92bc35d062c83f89b7ea87d99dca9 = middlewaresi } return https://www./link/d3d92bc35d062c83f89b7ea87d99dca9ttp.https://www./link/d3d92bc35d062c83f89b7ea87d99dca9andlerFunc(func(w https://www./link/d3d92bc35d062c83f89b7ea87d99dca9ttp.ResponseWriter, r *https://www./link/d3d92bc35d062c83f89b7ea87d99dca9ttp.Request) { if err := https://www./link/d3d92bc35d062c83f89b7ea87d99dca9(w, r, r.Context()); err != nil { https://www./link/d3d92bc35d062c83f89b7ea87d99dca9ttp.Error(w, err.Error(), https://www./link/d3d92bc35d062c83f89b7ea87d99dca9ttp.StatusInternalServerError) } }) }
// 使用 https://www./link/d3d92bc35d062c83f89b7ea87d99dca9andler := Chttps://www./link/d3d92bc35d062c83f89b7ea87d99dca9ain(https://www./link/d3d92bc35d062c83f89b7ea87d99dca9andleUser, Withttps://www./link/d3d92bc35d062c83f89b7ea87d99dca9Authttps://www./link/d3d92bc35d062c83f89b7ea87d99dca9, Withttps://www./link/d3d92bc35d062c83f89b7ea87d99dca9RateLimit) https://www./link/d3d92bc35d062c83f89b7ea87d99dca9ttp.ListenAndServe(":8080", https://www./link/d3d92bc35d062c83f89b7ea87d99dca9andler)
注意中间件顺序从右往左应用(即 Withttps://www./link/d3d92bc35d062c83f89b7ea87d99dca9RateLimit 最先执行),这是函数组合的自然特性。如果顺序写反,认证可能在限流之后才触发,导致未授权请求仍消耗配额。
最易被忽略的一点:所有中间件必须显式将 ctx 传递给下游,否则 cancel() 或超时信号无法穿透整条链。一旦某个环节忘了传 ctx,整条链就退化成阻塞式调用。
# java
# go
# golang
# 处理器
# ai
# 注册表
# 标准库
# 为什么
相关栏目:
【
网站优化151355 】
【
网络推广146373 】
【
网络技术251813 】
【
AI营销90571 】
相关推荐:
香港服务器网站卡顿?如何解决网络延迟与负载问题?
JavaScript Ajax实现异步通信
在Oracle关闭情况下如何修改spfile的参数
深圳网站制作公司好吗,在深圳找工作哪个网站最好啊?
Laravel如何与Docker(Sail)协同开发?(环境搭建教程)
使用豆包 AI 辅助进行简单网页 HTML 结构设计
极客网站有哪些,DoNews、36氪、爱范儿、虎嗅、雷锋网、极客公园这些互联网媒体网站有什么差异?
手机网站制作与建设方案,手机网站如何建设?
Laravel怎么使用artisan命令缓存配置和视图
Python文件操作最佳实践_稳定性说明【指导】
logo在线制作免费网站在线制作好吗,DW网页制作时,如何在网页标题前加上logo?
Laravel如何使用Spatie Media Library_Laravel图片上传管理与缩略图生成【步骤】
如何在Windows环境下新建FTP站点并设置权限?
网易LOFTER官网链接 老福特网页版登录地址
Laravel怎么处理异常_Laravel自定义异常处理与错误页面教程
html5的keygen标签为什么废弃_替代方案说明【解答】
html文件怎么打开证书错误_https协议的html打开提示不安全【指南】
太平洋网站制作公司,网络用语太平洋是什么意思?
JavaScript模板引擎Template.js使用详解
宙斯浏览器视频悬浮窗怎么开启 边看视频边操作其他应用教程
如何在阿里云域名上完成建站全流程?
php在windows下怎么调试_phpwindows环境调试操作说明【操作】
如何用AI帮你把自己的生活经历写成一个有趣的故事?
什么是javascript作用域_全局和局部作用域有什么区别?
绝密ChatGPT指令:手把手教你生成HR无法拒绝的求职信
魔方云NAT建站如何实现端口转发?
郑州企业网站制作公司,郑州招聘网站有哪些?
C++时间戳转换成日期时间的步骤和示例代码
音乐网站服务器如何优化API响应速度?
深圳网站制作的公司有哪些,dido官方网站?
Laravel怎么判断请求类型_Laravel Request isMethod用法
如何在万网主机上快速搭建网站?
如何做网站制作流程,*游戏网站怎么搭建?
JavaScript如何实现继承_有哪些常用方法
如何用ChatGPT准备面试 模拟面试问答与职场话术练习教程
JS经典正则表达式笔试题汇总
敲碗10年!Mac系列传将迎来「触控与联网」双革新
网站建设要注意的标准 促进网站用户好感度!
简单实现Android验证码
python中快速进行多个字符替换的方法小结
Laravel如何生成URL和重定向?(路由助手函数)
标准网站视频模板制作软件,现在有哪个网站的视频编辑素材最齐全的,背景音乐、音效等?
制作公司内部网站有哪些,内网如何建网站?
如何在腾讯云服务器上快速搭建个人网站?
制作无缝贴图网站有哪些,3dmax无缝贴图怎么调?
Laravel如何实现数据导出到CSV文件_Laravel原生流式输出大数据量CSV【方案】
Laravel如何配置.env文件管理环境变量_Laravel环境变量使用与安全管理
如何快速建站并高效导出源代码?
如何在万网自助建站平台快速创建网站?
高性能网站服务器部署指南:稳定运行与安全配置优化方案

