errors.Is判断错误链中是否存在目标哨兵错误,errors.As提取底层错误类型;二者配合使用可穿透多层包装,但需先判空、优先用预定义哨兵错误,避免对临时错误实例调用Is。
errors.Is 用于判断一个错误链中是否存在与目标错误相等的错误(基于 == 或实现了 Is(error) 方法)。它不关心错误类型,只关心“是不是同一个错误实例”或“是否被标记为相等”。
常见错误现象:用 errors.Is(err, io.EOF) 正确;但用 errors.Is(err, fmt.Errorf("not found")) 几乎总返回 false,因为每次 fmt.Errorf 都新建一个地址不同的错误实例。
io.EOF、os.ErrNotExist)或你自己显式定义的变量错误使用 errors.Is
errors.New / fmt.Errorf 调用结果)做 errors.Is 判断errors.As
errors.As 用于从错误链中向下查找第一个能被赋值给指定类型的错误值,并将其实例存入目标变量。它解决的是“这个错误底层是不是某种具体类型”的问题。
典型使用场景:你收到一个 os.PathError 包裹

Err 字段看是不是 syscall.EACCES;或者处理 net.OpError 时想提取底层的 syscall.Errno。
*os.PathError),errors.As 才能写入errors.As 只取第一个匹配的syscall.Errno 实现了 error,但它不是结构体指针,不能用 *syscall.Errno 接收,而要用 **syscall.Errno 或更稳妥地先转成 syscall.Errno 再比较示例:
var pathErr *os.PathError
if errors.As(err, &pathErr) {
log.Printf("path: %s, op: %s", pathErr.Path, pathErr.Op)
}
直接用 err == io.EOF 在简单场景可行,但一旦错误被 fmt.Errorf("wrap: %w", err) 或 errors.Wrap(旧库)包裹,就失效了——因为外层错误不是 io.EOF 本身。
switch err.(type) 只能匹配最外层错误类型,无法穿透包装。比如 fmt.Errorf("read failed: %w", os.ErrNotExist) 的类型是 *fmt.wrapError,不是 *os.PathError,更不是 os.ErrNotExist 的底层类型。
errors.Is 和 errors.As 是 Go 1.13+ 错误链标准方案,专为多层包装设计errors.Unwrap 实现类似逻辑非常容易漏层或 panic,不推荐github.com/pkg/errors)在 Go 1.13+ 后基本可弃用,因其 Causer / Unwrapper 接口已被标准库覆盖真实错误往往既需要判断“是不是某个哨兵错误”,又需要提取“底层是不是某种结构体”。二者不互斥,但有逻辑先后:通常先 errors.Is 快速识别已知错误,再用 errors.As 做精细处理。
容易踩的坑:在未确认错误非 nil 的情况下直接传给 errors.As,会导致 panic(因内部对 nil 解引用)。虽然标准库文档没明说,但实测 errors.As(nil, &x) 返回 false 不 panic;不过为保险起见,仍建议显式判空。
err != nil,再调用 errors.Is 或 errors.As
errors.Is(err, someSentinel) 为 true,通常无需再 As ——除非你要访问该哨兵错误的字段(但哨兵错误一般无字段)errors.As(err, &x) 成功,可进一步对 x 做 errors.Is(x.Err, ...) 判断其内部错误复杂点在于错误链可能深达 5–6 层,且中间混用标准库和旧式包装;这时候靠肉眼调试几乎不可行,建议配合 fmt.Printf("%+v", err) 查看完整链路。