不能直接用 errors.New 包装业务错误,因其仅返回固定字符串错误,无法携带状态码、trace_id、原始错误等上下文,导致分类处理、日志追踪和HTTP状态码映射失效。
errors.New 包装业务错误信息因为 errors.New 只返回一个带固定字符串的 error 实例,无法携带状态码、请求ID、原始错误等上下文。当需要做错误分类处理(比如重试、告警、前端提示)时,光靠 Error() 方法返回的字符串很难可靠判断。
"record not found"
trace_id 或 user_id
Go 的 error 接口只要求实现 Error() string 方法,但你可以额外添加方法(比如 Code()、TraceID()),只要类型满足接口即可。关键是:不要让自定义 error 实现 Unwrap() 除非你真需要链式错误(否则会干扰 errors.Is 和 errors.As 的行为)。
type AppError struct {
Code int
Message string
TraceID string
Err error // 原始底层错误,可选
}
func (e *AppError) Error() string {
return e.Message
}
func (e *AppError) Code() int {
return e.Code
}
func (e *AppError) TraceID() string {
return e.TraceID
}
*AppError),避免值拷贝丢失字段errors.Unwrap(),需显式实现该方法并返回 e.Err;否则别加Error() 中拼接 e.Err.Error() —— 这会让日志重复且难以解析errors.Is 和 errors.As 正确识别你的错误Go 标准库的

Unwrap() 链和类型断言。如果你的自定义 error 没实现 Unwrap(),errors.Is 就不会向下查找嵌套错误;而 errors.As 要求目标变量是指针类型才能成功赋值。
sql.ErrNoRows),必须实现 Unwrap() error 方法errors.As(err, &target) 时,target 必须是 *AppError 类型的变量Unwrap() 中返回非 error 类型或 nil,否则会导致 panic 或逻辑错乱func (e *AppError) Unwrap() error {
return e.Err
}
// 使用示例
var appErr *AppError
if errors.As(err, &appErr) {
log.Printf("app error code: %d, trace: %s", appErr.Code(), appErr.TraceID())
}
不要在每个 handler 里写 if err != nil { ... },而是用中间件或封装后的 Result 类型统一处理。重点是:只对实现了特定接口(如 StatusCode() int)的 error 做特殊响应,其余走 500。
fmt.Sprintf("%T", err) 判断类型——性能差且不可靠errors.As 提取业务 error,再调其方法获取状态码%+v(来自 github.com/pkg/errors 或 Go 1.19+ 的 fmt.Errorf("%w", err))保留栈信息复杂点在于:同一类错误可能来自不同层级(DB、RPC、校验),它们的 Code() 含义必须收敛且文档化。否则前端拿到 409 不知道是并发冲突还是资源已存在——这个一致性得靠团队约定和代码审查来守住。