Go函数必须将error作为最后一个返回值,这是标准约定;应使用fmt.Errorf加%w包装错误以保留上下文,避免硬编码字符串;需根据错误类型选择重试、提示或告警等处理方式。
Go 语言中函数返回 error 类型不是“可选技巧”,而是标准约定——几乎所有 I/O、解析、网络、文件操作等可能失败的函数,都以 error 作为最后一个返回值。不检查它,就等于默认忽略失败。
error 放在返回值最后?这是 Go 的惯用法(idiom),由标准库和社区共同固化。编译器不强制,但工具链(如 go vet)、linter(如 errcheck)和 IDE 都基于这个位置做静态分析。如果把它放在前面或中间,调用方用 if err != nil 判断时会破坏多值赋值的可读性,也容易引发漏判。
常见错误现象:
- 写成 func ReadFile() (error, []byte) → 调用时要写 err, data := ReadFile(),逻辑主次颠倒
- 忘记接收 error:直接 data := ReadFile() 导致编译失败(多值赋值未全接收)
正确写法示例:
func ReadConfig(path string) (map[string]string, error) {
data, err := os.ReadFile(path)
if err != nil {
return nil, fmt.Errorf("failed to read %s: %w", path, err)
}
return parseConfig(data), nil
}error?不要用 errors.New("xxx") 硬编码字符串就完事——它无法携带上下文、无法判断类型、不利于调试。优先用 fmt.Errorf 加 %w 包装,保留原始错误链。
fmt.Errorf("xxx: %w", err) —— 后续可用 errors.Is() 或 errors.As() 检查errors.New("xxx") 或 fmt.Errorf("xxx")(无 %w)error 接口的 struct,并实现 Error() string
反例:return errors.New("open failed") —— 完全丢失路径、权限、系统 errno 等关键信息。
error 返回值?别跳过判断,也别只打印日志就继续执行。处理方式取决于错误性质和业务场景:
err.Error() 和 fmt%+v),通知运维errors.Is(err, fs.ErrNotExist) 做精确匹配,而不是 strings.Contains(err.Error(), "no such file")
常见陷阱:
- 写 if err != nil { log.Println(err); return },但没返回具体错误给上层,导致调用方以为成功
- 在 defer 中覆盖已返回的 error(比如 defer func() { if f != nil { f.Close() } }() 里没检查 f.Close() 的 error)
error 和 panic 不是一个东西panic 是程序异常中断,用于真正不可恢复的编程错误(

error 是预期之内的失败路径,是正常控制流的一部分。把本该返回 error 的情况改成 panic,会让调用方失去处理能力,也违背 Go 的显式错误哲学。
一个典型误用:json.Unmarshal([]byte(`{`), &v) 应该返回 error,而不是让上层靠 recover 捕获 panic —— 这会让错误传播变得不可控,且无法做类型判断或重试。
真正难的不是写 if err != nil,而是想清楚这个错误发生时,当前函数该返回什么、是否该包装、调用方有没有能力/责任处理它。很多 bug 其实源于错误被静默吞掉,或者被过度包装丢了原始信息。