Go中判断error类型常用方法包括:1. 类型断言用于已知具体错误类型;2. errors.As推荐用于解包错误链并匹配具体类型;3. errors.Is用于比较预定义错误值;4. 自定义错误可通过As或类型断言处理,优先使用errors.As以支持错误包装。

在Go语言中,error 是一个接口类型,通常用于表示函数执行过程中发生的错误。由于 error 是接口,实际的错误值可能是各种具体类型。因此,在处理错误时,有时需要判断其具体类型以便做出不同响应。以下是几种常见的判断 error 类型的方法。
1. 使用类型断言(Type Assertion)
当你知道某个 error 可能是特定类型时,可以直接使用类型断言来判断。
例如,标准库中的某些错误有具体结构体类型:示例:
if err := someFunc(); err != nil {
if e, ok := err.(*os.PathError); ok {
fmt.Println("路径错误:", e.Path)
} else {
fmt.Println("其他错误:", err)
}
}
这里通过 e, ok := err.(*os.PathError) 判断 err 是否为 *os.PathError 类型。如果是,ok 为 true,并可访问其字段如 Path。
2. 使用 errors.As()(推荐方式)
从 Go 1.13 开始,官方推荐使用 errors.As() 来判断 error 是否属于某具体类型,它能递归地解包错误链,找到匹配的底层错误。
立即学习“go语言免费学习笔记(深入)”;
适用场景: 错误被包装过(比如用 fmt.Errorf 包装并添加上下文)。
示例:
err := readFile()
var pathErr *os.PathError
if errors.As(err, &pathErr) {
fmt.Println("发生路径错误,路径为:", pathErr.Path)
} else {
fmt.Println("不是路径错误")
}
即使 err 是像 wrapped: failed to open file: no such file or directory 这样的包装错误,只要底层包含 *os.PathError,errors.As 就能提取出来。
3. 判断是否为预定义错误值(errors.Is)
有些函数返回的是固定的错误变量(如 io.EOF、os.ErrNotExist),这时应使用 errors.Is() 来比较。
示例:
if errors.Is(err, os.ErrNotExist) {
fmt.Println("文件不存在")
}
这比直接用 == 更安全,因为 errors.Is 也会递归检查错误链中是否包含目标错误。
4. 自定义错误类型判断
如果你自己定义了错误类型,也可以用类型断言或 errors.As 来判断。
定义自定义错误:
type MyError struct {
Msg string
}
func (e *MyError) Error() string {
return e.Msg
}
判断类型:
if myErr, ok := err.(*MyError); ok {
fmt.Println("自定义错误信息:", myErr.Msg)
}
或者使用:
var myErr *MyError
if errors.As(err, &myErr) {
fmt.Println("捕获自定义错误:", myErr.Msg)
}
推荐使用 errors.As,因为它兼容包装后的错误。
基本上就这些。日常开发中,优先使用 errors.Is 比较已知错误值,用 errors.As 提取特定类型的错误。避免对 error 字符串进行判断,容易出错且不健壮。










