链式调用中error不可忽略,必须每步检查:DoA()→检查err→DoB()→检查err→DoC()→检查err;否则非法状态值可能导致panic或未定义行为。
Go 语言里链式调用(比如 DoA().DoB().DoC())本身不自带错误传播机制,一旦中间某步返回 error,后续方法若继续执行,大概率会 panic 或产生未定义行为。最常见错误是只检查最后一步的 error,而忽略了前序步骤失败后仍调用了后续方法。
标准做法是让每一步都返回 (T, error),上一步的 T 作为下一步的输入,同时在每步开头检查前序 error。这不是语法糖,而是显式控制流。
obj.DoA().DoB().DoC() 这类纯链式,除非所有方法都明确声明「不失败」或已做内部错误兜底res, err := New().DoA()
if err != nil {
return err
}
res, err = res.DoB()
if err != nil {
return err
}
_, err = res.DoC()
return errThen),但底层仍是逐个检查 error,不是魔法泛型能帮你抽象出通用的错误短路逻辑,但不能绕过「检查每个返回值」这一原则。
func Then[T a
ny](val T, fn func(T) (T, error)) (T, error),它只是把 if err != nil 封装了一层,调用时仍要处理最终的 error
fn 若本身没检查入参有效性,依然可能 panicdefer 和 recover 是针对 panic 的,不是 error 处理机制。把 error 转成 panic 再 recover,属于反模式。
error 是值,设计初衷就是显式传递和判断;转 panic 会丢失原始上下文(比如哪一步返回了什么 error)panic,会导致无法区分是程序员失误(如 nil 指针解引用)还是业务失败(如网络超时),增加排查成本sql.DB, http.Client)全部遵循显式 error 返回,模仿它们最稳妥error 后是否立刻响应——哪怕只写一行 if err != nil { return err },也比“先链完再统一处理”可靠得多。最容易被忽略的是:**中间步骤返回非 nil error 后,其返回值(如 struct 实例)很可能处于非法状态,此时再拿它调用任何方法,结果不可预测**。