答案:Go中推荐使用errors.Is判断语义等价,errors.As提取具体错误类型。二者支持错误链穿透,优于直接比较或类型断言。自定义错误应实现Unwrap方法以兼容标准库。

在 Golang 中,错误处理是程序健壮性的关键环节。由于 Go 没有异常机制,所有错误都通过返回 error 类型值来表示。实际开发中,我们常常需要判断错误的具体类型,以便做出不同的响应。本文将介绍几种常见的错误类型判断方法,并结合实践给出推荐做法。
使用 errors.Is 进行语义等价判断
从 Go 1.13 开始,标准库引入了 errors.Is 函数,用于判断两个错误是否表示相同的语义错误。它能穿透包装的错误(wrapped errors),适合判断“是否是某个特定错误”。
例如,你调用一个函数,它可能返回 os.ErrNotExist,即使这个错误被包装过,也能正确识别:
- if errors.Is(err, os.ErrNotExist) {
- fmt.Println("文件不存在")
- }
这种方式比直接比较更安全,支持错误链中的深层匹配。
立即学习“go语言免费学习笔记(深入)”;
使用 errors.As 提取特定错误类型
当你需要访问错误的具体结构字段或方法时,应使用 errors.As。它可以将错误链中的某一层提取到指定类型的变量中。
比如自定义了一个网络超时错误:
- type TimeoutError struct{}
- func (e *TimeoutError) Error() string { return "timeout" }
在处理错误时可以这样判断并获取实例:
- var te *TimeoutError
- if errors.As(err, &te) {
- fmt.Println("发生超时错误")
- // 可以调用 te 的其他方法或访问字段
- }
这比类型断言更安全,因为它会遍历整个错误链。
避免直接类型断言和字符串比较
有些人习惯用 err.Error() == "some message" 来判断错误,这种做法非常脆弱。错误信息可能变化,也可能被多层包装导致内容不一致。
同样,直接使用类型断言如 err.(*MyError) 在错误被包装后会失败,且无法处理嵌套情况。
推荐始终优先使用 errors.Is 和 errors.As,它们是 Go 官方为现代错误处理设计的标准方式。
自定义错误建议实现 Unwrap 方法
如果你自己实现错误包装,建议实现 Unwrap() error 方法,这样你的错误就能与 errors.Is 和 errors.As 兼容。
示例:
- type MyError struct {
- Msg string
- Err error
- }
- func (e *MyError) Error() string { return e.Msg }
- func (e *MyError) Unwrap() error { return e.Err }
这样上层代码就可以正常使用 errors.As 或 errors.Is 穿透你的错误包装。
基本上就这些。Go 的错误判断不再需要繁琐的类型切换,合理使用标准库提供的工具能让代码更清晰、更可靠。关键是理解错误的语义层次,选择合适的方法进行判断。










