Go中判断错误类型应按场景选择:精确匹配未包装错误用类型断言;检查错误链中是否存在某类型用errors.As;判断是否等于哨兵错误用errors.Is。
在 Go 中判断错误类型,核心是区分“是否为某类错误”和“是否包含某类错误”。类型断言适合精确匹配具体错误类型(如 *os.PathError),而 errors.As 更适用于检查错误链中是否存在某个目标类型的错误(支持包装错误,如用 fmt.Errorf("xxx: %w", err) 包装的情况)。
当明确知道错误变量可能直接是某个具体类型(未被包装)时,可用类型断言:
err.(*os.PathError) 或更安全的双返回值形式:if pe, ok := err.(*os.PathError); ok { ... }
%w 包装过(如 fmt.Errorf("open failed: %w", pe)),断言会失败*os.PathEr
ror、*net.OpError、*exec.ExitError
errors.As 会沿着错误链(通过 Unwrap())逐层查找,只要某一层是目标类型或实现了该接口,就返回 true:
var pe *os.PathError; if errors.As(err, &pe) { ... }(注意传入指针地址)fmt.Errorf("read: %w", fmt.Errorf("io: %w", pe)) 也能成功匹配 *os.PathError
interface{ Timeout() bool }
如果要判断错误是否等于某个预定义的错误变量(如 os.ErrNotExist),应使用 errors.Is:
if errors.Is(err, os.ErrNotExist) { ... }== 目标值,适合哨兵错误(sentinel errors)== 直接比较,因为包装后的错误地址不同优先按场景选择:
errors.As
sql.ErrNoRows)?→ 用 errors.Is
errors.As 和 errors.Is