Golang中的错误处理与面向对象思想 Go语言错误作为第一等公民

6次阅读

goError 是返回值而非异常,需显式处理;应避免嵌套 if err != nil,优先用 %w 包装、errors.is/as 判断,http 错误需脱敏并结构化处理。

Golang中的错误处理与面向对象思想 Go语言错误作为第一等公民

Go 里 error 不是异常,别用 try-catch 思维写代码

Go 没有 throwcatchfinallyerror 是普通接口类型,不是控制流机制。很多人从 Java/Python 转来,下意识想“捕获错误”,结果写出一嵌套 if err != nil 还以为是“优雅处理”。

真正该做的是:把错误当返回值处理,像处理 intString 一样自然。函数签名里显式声明它,调用方必须面对它——这不是限制,是强制你思考失败路径。

  • error 接口只有 Error() string 方法,所以自定义错误只需实现这个方法,不用继承任何基类
  • 不要在函数里 log.Fatalos.Exit ——这等于把错误处理权抢走,上层无法恢复或重试
  • 常见错误:用 fmt.Errorf("failed: %v", err) 丢掉原始错误类型和堆;应优先用 fmt.Errorf("failed: %w", err)(带 %w)保留包装关系

什么时候该用 errors.Is,什么时候用 errors.As

Go 1.13 引入的这两个函数,解决的是“怎么判断一个 error 是不是我关心的那种错误”。直接用 ==strings.Contains(err.Error(), "...") 都不可靠——前者忽略包装,后者脆弱且无法跨语言本地化。

  • errors.Is(err, fs.ErrNotExist):判断是否是某个已知错误(或其包装链中的任意一层),适合做“存在性检查”场景,比如文件不存在就创建
  • errors.As(err, &target):尝试把 err 解包成具体类型,适合需要访问错误内部字段时,比如解析 net.OpError 获取 Addr 字段
  • 注意:errors.As 第二个参数必须是指针,传值会 panic;而 errors.Is 的第二个参数不能是 nil,否则行为未定义

自定义错误类型别乱加字段,先想清楚要不要导出

很多人一上来就写 type MyError Struct { Code int; Message string; ReqID string },然后所有地方都 &MyError{...} 返回。问题在于:这些字段是不是真的要暴露给调用方?有没有可能被误用或强依赖?

立即学习go语言免费学习笔记(深入)”;

  • 如果只是日志记录或调试用,用 fmt.Errorf("timeout for %s: %w", op, cause) + %w 包装更轻量,也不泄露内部结构
  • 如果必须结构化(比如 gRPC 错误码映射),字段名尽量小写(不导出),只通过方法暴露必要信息,例如 func (e *myError) Code() int
  • 别为了“面向对象感”而实现一堆 getter/setter;Go 的错误类型不是用来建模业务状态的,它是失败信号的载体

HTTP handler 里别把 error 直接塞进 http.Error

开发 Web API 时,看到 http.Error(w, err.Error(), http.StatusInternalServerError) 就该警觉。这等于把底层错误细节(比如数据库连接失败、sql 语法错)全扔给前端,既不安全,也不利于客户端分类处理。

  • 应该用中间层统一转换:把底层 error 映射为预定义的业务错误码(如 ErrUserNotFound),再转成 json 响应体
  • 开发阶段可保留原始错误(加 X-Debug-Error header),生产环境必须脱敏;err.Error() 永远不能直接返回给用户
  • 别在 handler 里用 log.printf 打印错误就完事——漏掉上下文(如请求 ID、参数)会让排查变成猜谜;建议用结构化日志库(如 zap)绑定 reqIDerr

最常被忽略的一点:错误不是越详细越好,而是要在“可诊断性”和“安全性/抽象层级”之间卡准位置。包装太多层会丢失关键上下文,不包装又难定位根因——得看调用链里谁真正需要知道这个细节。

text=ZqhQzanResources