通过接口和结构体定义错误类型,结合errors.As和错误包装,可构建可识别的Go错误分类体系。

在Go语言中,错误处理是程序健壮性的关键部分。虽然
error
为了实现可区分的错误分类,建议基于业务或系统层级定义不同的错误类型。可以通过接口来抽象错误类别,让具体错误类型实现这些接口,从而实现类型断言区分。
例如,定义几个常见错误分类接口:
type NetworkError interface {
IsNetwork() bool
}
type ValidationError interface {
IsValidation() bool
}
type AuthError interface {
IsAuth() bool
}
然后让具体的错误类型实现这些接口,调用方即可通过类型断言判断错误类别:
立即学习“go语言免费学习笔记(深入)”;
<pre class="brush:php;toolbar:false;">if err, ok := err.(interface{ IsNetwork() bool }); ok {
// 处理网络错误
}
比起返回简单的字符串错误,使用结构体能携带更多信息,并支持类型比较。定义不同的错误结构体,可以自然形成分类体系。
示例:
type ValidationError struct {
Field string
Message string
}
func (e *ValidationError) Error() string {
return fmt.Sprintf("validation error on field %s: %s", e.Field, e.Message)
}
func (e *ValidationError) IsValidation() bool { return true }
type NetworkError struct {
Op string
Err error
}
func (e *NetworkError) Error() string {
return fmt.Sprintf("network error during %s: %v", e.Op, e.Err)
}
func (e *NetworkError) IsNetwork() bool { return true }
这样在调用侧可以通过
errors.As
var netErr *NetworkError
if errors.As(err, &netErr) {
log.Printf("network failure: %v", netErr)
}
Go 1.13+ 提供了
errors.Unwrap
errors.Is
errors.As
建议在错误传递过程中使用
fmt.Errorf
%w
if err != nil {
return fmt.Errorf("failed to process user: %w", err)
}
这样外层仍可通过
errors.As
为避免散落的错误构造逻辑,可定义错误工厂函数,统一创建特定类型的错误:
func NewValidationError(field, msg string) *ValidationError {
return &ValidationError{Field: field, Message: msg}
}
func NewNetworkError(op string, err error) *NetworkError {
return &NetworkError{Op: op, Err: err}
}
这样不仅提升代码一致性,也便于未来扩展错误元数据(如错误码、日志标签等)。
基本上就这些。通过接口定义分类、结构体承载细节、包装保留上下文、工厂统一构造,就能构建出清晰可维护的Go错误类型体系。关键在于让错误“可识别”而不仅是“可读”。
以上就是如何实现Golang的错误分类 创建可区分的错误类型体系的详细内容,更多请关注php中文网其它相关文章!
每个人都需要一台速度更快、更稳定的 PC。随着时间的推移,垃圾文件、旧注册表数据和不必要的后台进程会占用资源并降低性能。幸运的是,许多工具可以让 Windows 保持平稳运行。
Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号