答案是:Go中应使用结构化错误替代字符串错误,通过定义含Code、Message、Details、Err的BizError实现可维护的错误体系,结合错误码常量、工厂函数、HTTP映射中间件和结构化日志,提升诊断效率与协作体验。

在 Go 语言中,业务错误不应只是 errors.New("xxx") 或 fmt.Errorf("xxx") 这样的字符串错误。真正可维护、可诊断、可扩展的错误体系,需要结构化封装——核心是定义一个带业务上下文、错误码、原始原因、日志友好字段的自定义 error struct。
定义标准业务错误结构体
推荐一个轻量但足够表达业务语义的结构:
- Code:int 类型错误码(如 1001 表示“用户不存在”,建议用常量管理)
- Message:面向开发/日志的简明描述(非前端展示用)
- Details:map[string]interface{} 或 struct{},存放调试信息(如请求 ID、参数快照、SQL 片段)
-
Err:嵌套原始 error(支持链式错误,兼容
errors.Is/errors.As)
实现 Error() 方法返回 Message,同时实现 Unwrap() 返回 Err,即可无缝接入 Go 1.13+ 错误链生态。
统一错误码与错误工厂函数
避免散落的 magic number 和重复构造。集中定义错误码常量,并提供工厂函数:
立即学习“go语言免费学习笔记(深入)”;
- 按模块分组常量,例如
user.ErrNotFound = 2001、order.ErrInvalidAmount = 3005 - 工厂函数如
user.NewNotFoundErr(uid) *BizError,内部自动填充 Code/Message/Details - 支持链式包装:
user.NewNotFoundErr(uid).Wrap(err),保留原始错误栈
HTTP 层错误映射与响应
不要在 handler 里手动 switch code。建议用中间件或统一响应封装:
- 定义
ErrorResponse结构,含 code(HTTP 状态码)、biz_code(业务码)、message、trace_id - 写一个
ResolveBizError(err error) *ErrorResponse函数,根据BizError.Code映射 HTTP 状态码(如 404→404,1001→404,500x→500) - handler 中直接
return handleError(ctx, err),无需每个地方判断
日志与可观测性增强
结构化错误天然适配 JSON 日志。关键点:
- 记录时调用
log.Error("biz failed", "err", bizErr, "details", bizErr.Details) - 在
BizError中加入TraceID string字段,从 context 注入,打通全链路追踪 - 对敏感字段(如手机号、token)在
Details中自动脱敏,或提供Redact()方法
基本上就这些。不复杂但容易忽略的是:坚持用结构体代替字符串错误、所有错误创建走工厂、HTTP 响应逻辑收口。做下来,排查效率和协作体验会明显提升。










