Golang Go语言中优雅的错误处理 支持错误码 错误堆栈 错误链的工具库
https://github.com/morrisxyang/errors
在业务开发中需要一个支持错误码对外返回, 堆栈打印等能力的错误工具库, 先开始使用 pkg/errors, 但该项目已经只读, 上次更新是几年前, 而且有一些点比如调整堆栈打印深度, 打印格式等能力没有支持, 后续根据业务的需要抽取了一个通用库, 做了一些优化, 详见下方.
如果觉得有用欢迎 Star 和 PR, 有问题请直接提 issue
errors
简单的支持错误堆栈, 错误码, 错误链的工具库:
-
支持携带堆栈, 嵌套构造错误链
-
支持携带错误码, 方便接口返回
-
支持自定义堆栈打印深度和错误链打印格式
-
使用 CallersFrames 替代 FuncForPC 生成堆栈, 避免特殊情况
line number 错误
等问题, 详见runtime: strongly encourage using CallersFrames over FuncForPC with Callers result -
简化堆栈信息, 一条链路多次
Wrap
操作只保留最深层堆栈, 只打印一次
安装和文档
安装使用 go get github.com/morrisxyang/errors
文档地址是 https://pkg.go.dev/github.com/morrisxyang/errors
快速开始
构造错误链
func a() error {
err := b()
err = Wrap(err, "a failed reason")
return err
}
func b() error {
err := c()
err = Wrap(err, “b failed reason”)
return err
}
func c() error {
_, err := os.Open(“test”)
if err != nil {
return WrapWithCode(err, 123, “c failed reason”)
}
return nil
}
打印错误信息, %+v
会打印堆栈, %v
只打印错误信息
a failed reason
Caused by: b failed reason
Caused by: 123, c failed reason
Caused by: open test: no such file or directory
github.com/morrisxyang/errors.c
/Users/morrisyang/Nutstore Files/go-proj/githuberrors/errors_test.go:94
github.com/morrisxyang/errors.b
/Users/morrisyang/Nutstore Files/go-proj/githuberrors/errors_test.go:86
github.com/morrisxyang/errors.a
/Users/morrisyang/Nutstore Files/go-proj/githuberrors/errors_test.go:80
....堆栈信息省略
核心方法
错误封装
- func New(msg string) error
- func Newf(format string, args ...interface{}) error
- func NewWithCode(code int, msg string) error
- func NewWithCodef(code int, format string, args ...interface{}) error
- func Wrap(e error, msg string) error
- func Wrapf(e error, format string, args ...interface{}) error
- func WrapWithCode(e error, code int, msg string) error
- func WrapWithCodef(e error, code int, format string, args ...interface{}) error
错误解析
- func Code(e error) int
- func EffectiveCode(e error) int
- func Msg(e error) string
- func As(err error, target interface{}) bool
- func Is(err, target error) bool
- func Cause(e error) error
- func Unwrap(err error) error
配置
FAQ
-
多次 Wrap 错误会携带多次堆栈吗?
可在调用链路上多次 Wrap, 添加说明信息, 但只有最深层的 Wrap 操作会设置堆栈, 继续
Wrap
,return err
等操作不会影响堆栈信息 -
在链路中某个错误设置了合适的错误码, 然后继续 Wrap 时没有设置, 如何获取?
建议在合适的清晰的时机设置有效的错误码, 可以使用
EffectiveCode
获取链路中外层第一个有效的非 0 错误码, 由于系统调用等情况, 同一链路中可能有多个错误携带错误码, 此时默认外层的错误码应该对外暴露, 屏蔽了内层的详细信息.
Golang Go语言中优雅的错误处理 支持错误码 错误堆栈 错误链的工具库
更多关于Golang Go语言中优雅的错误处理 支持错误码 错误堆栈 错误链的工具库的实战系列教程也可以访问 https://www.itying.com/category-94-b0.html
标准库里面内置的就有 errors 啊
更多关于Golang Go语言中优雅的错误处理 支持错误码 错误堆栈 错误链的工具库的实战系列教程也可以访问 https://www.itying.com/category-94-b0.html
标准库的 errors 没有堆栈, 这里主要是增加了堆栈, 以及微服务常用的错误码等支持
啊, 确实, 我知道了. 我想想
在Golang中,优雅的错误处理是确保应用程序健壮性和可靠性的关键。对于支持错误码、错误堆栈和错误链的工具库,以下是一些建议:
-
自定义错误类型:
- Go语言本身不直接支持错误码,但可以通过自定义错误类型来实现。例如,定义一个包含错误码和错误信息的结构体,并实现
error
接口。
- Go语言本身不直接支持错误码,但可以通过自定义错误类型来实现。例如,定义一个包含错误码和错误信息的结构体,并实现
-
错误堆栈:
- Go语言的标准库提供了获取错误堆栈的功能,通常通过
panic
和recover
机制,以及runtime
包中的函数来实现。在捕获异常时,可以使用debug.PrintStack()
来打印堆栈信息。
- Go语言的标准库提供了获取错误堆栈的功能,通常通过
-
错误链:
- 从Go 1.13版本开始,标准库
errors
提供了Unwrap
接口和fmt.Errorf
的%w
格式化动词,用于构建和解析错误链。这有助于在多层代码中传播错误时保留完整的上下文信息。
- 从Go 1.13版本开始,标准库
-
工具库:
- 对于更高级的错误处理需求,可以考虑使用第三方库,如
pkg/errors
。该库提供了更丰富的错误处理功能,如堆栈跟踪、错误链和格式化错误信息,有助于在复杂的应用程序中进行有效的错误诊断和处理。
- 对于更高级的错误处理需求,可以考虑使用第三方库,如
综上所述,通过自定义错误类型、利用标准库的错误堆栈功能、使用错误链以及借助第三方库,可以在Golang中实现优雅的错误处理。