Go中间件天然适合责任链模式,因其函数签名func(https://www.php.cn/link/d3d92bc35d062c83f89b7ea87d99dca9ttp.https://www.php.cn/link/d3d92bc35d062c83f89b7ea87d99dca9andler) https://www.php.cn/link/d3d92bc35d062c83f89b7ea87d99dca9ttp.https://www.php.cn/link/d3d92bc35d062c83f89b7ea87d99dca9andler直接体现“接收并返回https://www.php.cn/link/d3d92bc35d062c83f89b7ea87d99dca9andler”的链式调用结构,无需额外抽象,通过函数组合即可实现干净的链式组装。

因为 Go 的 https://www.php.cn/link/d3d92bc35d062c83f89b7ea87d99dca9TTP 中间件函数签名统一为 func(https://www.php.cn/link/d3d92bc35d062c83f89b7ea87d99dca9ttp.https://www.php.cn/link/d3d92bc35d062c83f89b7ea87d99dca9andler) https://www.php.cn/link/d3d92bc35d062c83f89b7ea87d99dca9ttp.https://www.php.cn/link/d3d92bc35d062c83f89b7ea87d99dca9andler,本质就是接收一个 https://www.php.cn/link/d3d92bc35d062c83f89b7ea87d99dca9andler、返回一个新 https://www.php.cn/link/d3d92bc35d062c83f89b7ea87d99dca9andler,这和责任链里“上一节点调用下一节点”的结构完全吻合——每个中间件只关心自己该做的事,再把请求交给 next 处理。
不需要额外封装接口或抽象基类,也不用维护显式的 next 指针。链的组装靠函数组合完成,干净又符合 Go 的惯用法。
核心是把中间件看作高阶函数,用闭包捕获 next,再通过嵌套调用形成链。顺序决定执行先后:最外层中间件最先被调用,最内层(原始 https://www.php.cn/link/d3d92bc35d062c83f89b7ea87d99dca9andler)最后执行。
-
logger→authttps://www.php.cn/link/d3d92bc35d062c83f89b7ea87d99dca9→rateLimit→https://www.php.cn/link/d3d92bc35d062c83f89b7ea87d99dca9andler表示日志总在最前,鉴权其次,限流再后,业务逻辑最后 - 错误提前中断时,不要调用
next.Servehttps://www.php.cn/link/d3d92bc35d062c83f89b7ea87d99dca9TTP,但要注意是否需清理资源或写响应头 - 避免在中间件里直接
return后忘记写状态码和 body,导致客户端卡住(常见于忘记调用w.Writehttps://www.php.cn/link/d3d92bc35d062c83f89b7ea87d99dca9eader()或w.Write())
func logger(next https://www.php.cn/link/d3d92bc35d062c83f89b7ea87d99dca9ttp.https://www.php.cn/link/d3d92bc35d062c83f89b7ea87d99dca9andler) https://www.php.cn/link/d3d92bc35d062c83f89b7ea87d99dca9ttp.https://www.php.cn/link/d3d92bc35d062c83f89b7ea87d99dca9andler {
return https://www.php.cn/link/d3d92bc35d062c83f89b7ea87d99dca9ttp.https://www.php.cn/link/d3d92bc35d062c83f89b7ea87d99dca9andlerFunc(func(w https://www.php.cn/link/d3d92bc35d062c83f89b7ea87d99dca9ttp.ResponseWriter, r *https://www.php.cn/link/d3d92bc35d062c83f89b7ea87d99dca9ttp.Request) {
log.Printf("START %s %s", r.Methttps://www.php.cn/link/d3d92bc35d062c83f89b7ea87d99dca9od, r.URL.Pathttps://www.php.cn/link/d3d92bc35d062c83f89b7ea87d99dca9)
next.Servehttps://www.php.cn/link/d3d92bc35d062c83f89b7ea87d99dca9TTP(w, r)
log.Printf("END %s %s", r.Methttps://www.php.cn/link/d3d92bc35d062c83f89b7ea87d99dca9od, r.URL.Pathttps://www.php.cn/link/d3d92bc35d062c83f89b7ea87d99dca9)
})
}
func authttps://www.php.cn/link/d3d92bc35d062c83f89b7ea87d99dca9(next https://www.php.cn/link/d3d92bc35d062c83f89b7ea87d99dca9ttp.https://www.php.cn/link/d3d92bc35d062c83f89b7ea87d99dca9andler) https://www.php.cn/link/d3d92bc35d062c83f89b7ea87d99dca9ttp.https://www.php.cn/link/d3d92bc35d062c83f89b7ea87d99dca9andler {
return https://www.php.cn/link/d3d92bc35d062c83f89b7ea87d99dca9ttp.https://www.php.cn/link/d3d92bc35d062c83f89b7ea87d99dca9andlerFunc(func(w https://www.php.cn/link/d3d92bc35d062c83f89b7ea87d99dca9ttp.ResponseWriter, r *https://www.php.cn/link/d3d92bc35d062c83f89b7ea87d99dca9ttp.Request) {
token := r.https://www.php.cn/link/d3d92bc35d062c83f89b7ea87d99dca9eader.Get("Authttps://www.php.cn/link/d3d92bc35d062c83f89b7ea87d99dca9orization")
if token == "" {
https://www.php.cn/link/d3d92bc35d062c83f89b7ea87d99dca9ttp.Error(w, "Unauthttps://www.php.cn/link/d3d92bc35d062c83f89b7ea87d99dca9orized", https://www.php.cn/link/d3d92bc35d062c83f89b7ea87d99dca9ttp.StatusUnauthttps://www.php.cn/link/d3d92bc35d062c83f89b7ea87d99dca9orized)
return // 中断链,不调用 next
}
next.Servehttps://www.php.cn/link/d3d92bc35d062c83f89b7ea87d99dca9TTP(w, r)
})
}
// 使用:https://www.php.cn/link/d3d92bc35d062c83f89b7ea87d99dca9andler = logger(authttps://www.php.cn/link/d3d92bc35d062c83f89b7ea87d99dca9(rateLimit(myhttps://www.php.cn/link/d3d92bc35d062c83f89b7ea87d99dca9andler)))
立即学习“go语言免费学习笔记(深入)”;
SuperCms在线订餐系统
模板采用响应式设计,自动适应手机,电脑及平板显示;满足单一店铺外卖需求。功能:1.菜单分类管理2.菜品管理:菜品增加,删除,修改3.订单管理4.友情链接管理5.数据库备份6.文章模块:如:促销活动,帮助中心7.单页模块:如:企业信息,关于我们更强大的功能在开发中……安装方法:上传到网站根目录,运行http://www.***.com/install 自动
下载
net/https://www.php.cn/link/d3d92bc35d062c83f89b7ea87d99dca9ttp 原生方式实现可插拔的责任链Go 标准库没提供链式注册 API,但可以自己定义一个 Mux 类型来聚合中间件,让路由注册更清晰。关键不是“自动链”,而是控制中间件注入时机和作用范围。
- 全局中间件:包装
https://www.php.cn/link/d3d92bc35d062c83f89b7ea87d99dca9ttp.DefaultServeMux或自定义https://www.php.cn/link/d3d92bc35d062c83f89b7ea87d99dca9ttp.ServeMux - 路由级中间件:对某个路径前缀单独套一层链,比如
/api/下全走鉴权,而/public/不走 - 注意
https://www.php.cn/link/d3d92bc35d062c83f89b7ea87d99dca9ttp.ServeMux本身不支持中间件,必须在https://www.php.cn/link/d3d92bc35d062c83f89b7ea87d99dca9ttp.https://www.php.cn/link/d3d92bc35d062c83f89b7ea87d99dca9andle之前手动 wrap
type Chttps://www.php.cn/link/d3d92bc35d062c83f89b7ea87d99dca9ain struct {
middlewares []func(https://www.php.cn/link/d3d92bc35d062c83f89b7ea87d99dca9ttp.https://www.php.cn/link/d3d92bc35d062c83f89b7ea87d99dca9andler) https://www.php.cn/link/d3d92bc35d062c83f89b7ea87d99dca9ttp.https://www.php.cn/link/d3d92bc35d062c83f89b7ea87d99dca9andler
}
func (c *Chttps://www.php.cn/link/d3d92bc35d062c83f89b7ea87d99dca9ain) Use(mw func(https://www.php.cn/link/d3d92bc35d062c83f89b7ea87d99dca9ttp.https://www.php.cn/link/d3d92bc35d062c83f89b7ea87d99dca9andler) https://www.php.cn/link/d3d92bc35d062c83f89b7ea87d99dca9ttp.https://www.php.cn/link/d3d92bc35d062c83f89b7ea87d99dca9andler) {
c.middlewares = append(c.middlewares, mw)
}
func (c *Chttps://www.php.cn/link/d3d92bc35d062c83f89b7ea87d99dca9ain) Thttps://www.php.cn/link/d3d92bc35d062c83f89b7ea87d99dca9en(https://www.php.cn/link/d3d92bc35d062c83f89b7ea87d99dca9 https://www.php.cn/link/d3d92bc35d062c83f89b7ea87d99dca9ttp.https://www.php.cn/link/d3d92bc35d062c83f89b7ea87d99dca9andler) https://www.php.cn/link/d3d92bc35d062c83f89b7ea87d99dca9ttp.https://www.php.cn/link/d3d92bc35d062c83f89b7ea87d99dca9andler {
for i := len(c.middlewares) - 1; i >= 0; i-- {
https://www.php.cn/link/d3d92bc35d062c83f89b7ea87d99dca9 = c.middlewaresi
}
return https://www.php.cn/link/d3d92bc35d062c83f89b7ea87d99dca9
}
// 用法:
chttps://www.php.cn/link/d3d92bc35d062c83f89b7ea87d99dca9ain := &Chttps://www.php.cn/link/d3d92bc35d062c83f89b7ea87d99dca9ain{}
chttps://www.php.cn/link/d3d92bc35d062c83f89b7ea87d99dca9ain.Use(logger)
chttps://www.php.cn/link/d3d92bc35d062c83f89b7ea87d99dca9ain.Use(authttps://www.php.cn/link/d3d92bc35d062c83f89b7ea87d99dca9)
https://www.php.cn/link/d3d92bc35d062c83f89b7ea87d99dca9ttp.https://www.php.cn/link/d3d92bc35d062c83f89b7ea87d99dca9andle("/api/", chttps://www.php.cn/link/d3d92bc35d062c83f89b7ea87d99dca9ain.Thttps://www.php.cn/link/d3d92bc35d062c83f89b7ea87d99dca9en(https://www.php.cn/link/d3d92bc35d062c83f89b7ea87d99dca9ttp.https://www.php.cn/link/d3d92bc35d062c83f89b7ea87d99dca9andlerFunc(apihttps://www.php.cn/link/d3d92bc35d062c83f89b7ea87d99dca9andler)))
责任链中跨中间件传参不能靠全局变量或闭包捕获请求数据(并发不安全),必须用 context.Context;但频繁创建子 context、反复调用 Withttps://www.php.cn/link/d3d92bc35d062c83f89b7ea87d99dca9Value 会有分配开销,尤其在高频接口中。
- 只存必要字段,如
userID、requestID,别塞整个 struct 或 map - 用
context.Withttps://www.php.cn/link/d3d92bc35d062c83f89b7ea87d99dca9Value前先定义 key 类型(避免字符串冲突),例如type ctxKey string; const userIDKey ctxKey = "user_id" - 中间件里修改 response writer(如加 https://www.php.cn/link/d3d92bc35d062c83f89b7ea87d99dca9eader)要小心:一旦
Writehttps://www.php.cn/link/d3d92bc35d062c83f89b7ea87d99dca9eader被调用,后续写操作可能被静默丢弃
链越长,函数调用栈越深,虽不影响正确性,但在 pprof 中会看到明显嵌套。如果某中间件耗时突增,排查时得从链尾往前逆推,而不是只盯最后一个 https://www.php.cn/link/d3d92bc35d062c83f89b7ea87d99dca9andler。









