判断自定义错误应优先用errors.Is(值匹配哨兵错误)或errors.As(类型匹配并提取结构体字段),类型断言仅适用于未包装的单层错误场景。

在 Go 中判断一个 error 是否为自定义错误,核心在于理解 Go 的错误机制:Go 的 error 是接口,自定义错误通常实现该接口,而类型匹配需结合 errors.Is、errors.As 和类型断言三种方式,具体用哪种取决于你的设计目标——是检查错误链中是否存在某个**特定值**(如哨兵错误),还是想提取某个**具体错误类型**的实例进行进一步操作。
当你定义了一个包级变量作为“哨兵错误”(如 var ErrNotFound = errors.New("not found")),应使用 errors.Is 判断它是否出现在错误链中。它会自动遍历 Unwrap() 链,适合做“是否发生了某类已知错误”的布尔判断。
if errors.Is(err, ErrNotFound) { return http.StatusNotFound } )bool
errors.Is,它只比对指针或可比较的值(如 errors.New 或 fmt.Errorf 带 %w 包装的哨兵)当你定义了带字段的自定义错误类型(如 type ValidationError struct { Field string; Msg string }),且实现了 Error() 和可选的 Unwrap() 方法,就该用 errors.As。它会在错误链中查找第一个匹配该类型的实例,并将其实例赋值给目标变量。
var ve ValidationError; if errors.As(err, &ve) { log.Printf("validation failed on %s: %s", ve.Field, ve.Msg) }
&ve),否则匹配失败err 是 fmt.Errorf("failed: %w", &ValidationError{...}),errors.As 仍能成功提取如果确定错误没有被包装(即不是通过 %w 包装的),且你只需要快速判断其原始类型,可用类型断言 err.(*MyError)。但这种方式不推荐用于通用错误处理,因为它无法穿透错误链,也不符合 Go 错误最佳实践。
assert.IsType(t, &MyError{}, err))err.(*MyError))或返回 nil(若用 err, ok := err.(*MyError)),需额外判空为了让 errors.Is 和 errors.As 正常工作,自定义错误类型最好遵循以下习惯:
var 定义(var ErrInvalid = errors.New("invalid input")),不要每次调用都 errors.New
Unwrap() error 方法(如果支持包装),返回内部嵌套的 errorerrors.As
基本上就这些。记住:用 errors.Is 回答“是不是这个错”,用 errors.As 回答“这个错里有没有我想要的类型”,类型断言只是临时辅助手段。不复杂但容易忽略细节。
以上就是Go中如何判断一个error是否为自定义错误_Go Error类型匹配方法解析的详细内容,更多请关注php中文网其它相关文章!
每个人都需要一台速度更快、更稳定的 PC。随着时间的推移,垃圾文件、旧注册表数据和不必要的后台进程会占用资源并降低性能。幸运的是,许多工具可以让 Windows 保持平稳运行。
Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号