Go中责任链的典型误用是硬套Java的Handler继承结构,正确做法是用函数链:定义Handler为func(context.Context, interface{}) (interface{}, error),通过闭包组合,中断靠返回非nil错误或nil,nil,且每个处理器需首行检查ctx.Done()响应取消。
很多人一上来就写 Handler 接口 + Next 字段,结果发现链路无法中断、错误处理混乱、中间件顺序难控。这不是 Go 风格的责任链 —— Go 没有继承体系,硬套 Java 的 AbstractHandler 模式只会让代码变重、测试变难。
Go 更自然的方式是把每个处理步骤定义为 func(context.Context, interface{}) (interface{}, error),然后用闭包组合。这样链路可拆、可测、可跳过,且无需维护 Next 指针。
nil, nil 表示终止后续type Handler func(context.Context, interface{}) (interface{}, error)
func Chain(hs ...Handler) Handler {
return func(ctx context.Context, req interface{}) (interface{}, error) {
for _, h := range hs {
res, err := h(ctx, req)
if err != nil {
return nil, err
}
if res == nil { // 显式终止
return nil, nil
}
req = res // 向下传递结果
}
return req, nil
}
}
直接用 interface{} 会丢失编译期检查,但为每种事件定义独立链又太碎。折中方案是:用泛型约束输入输出类型,但链本身仍保持函数签名统一。
AuthHandler[T any]
Handler 类型,靠调用方保证类型流正确req.(T) + ok 判断,不 panic常见错误:在 Chain 内部对 req 做强制类型转换,导致运行时报 panic: interface conversion。
责任链不是孤立执行的,每个环节都该响应 ctx.Done()。别等整个链跑完才发现超时 —— 要在每个处理器开头检查。
Handler 必须第一行写 select { case
ctx,例如 db.QueryContext(ctx, ...)
容易被忽略的是:如果某个处理器内部启动了子 goroutine 并未绑定 ctx,取消信号就失效了。这种“漏网 goroutine”会让服务 hang 住。