0

0

如何使用Golang的errors.Is与errors.As判断错误类型_Golang错误判断技巧

P粉602998670

P粉602998670

发布时间:2026-01-30 15:53:07

|

989人浏览过

|

来源于php中文网

原创

errors.Is判断错误链中是否存在目标哨兵错误,errors.As提取底层错误类型;二者配合使用可穿透多层包装,但需先判空、优先用预定义哨兵错误,避免对临时错误实例调用Is。

如何使用golang的errors.is与errors.as判断错误类型_golang错误判断技巧

errors.Is 判断错误是否等于某个目标错误

errors.Is 用于判断一个错误链中是否存在与目标错误相等的错误(基于 == 或实现了 Is(error) 方法)。它不关心错误类型,只关心“是不是同一个错误实例”或“是否被标记为相等”。

常见错误现象:用 errors.Is(err, io.EOF) 正确;但用 errors.Is(err, fmt.Errorf("not found")) 几乎总返回 false,因为每次 fmt.Errorf 都新建一个地址不同的错误实例。

  • 只对预定义的哨兵错误(如 io.EOFos.ErrNotExist)或你自己显式定义的变量错误使用 errors.Is
  • 不要对临时构造的错误(errors.New / fmt.Errorf 调用结果)做 errors.Is 判断
  • 若错误来自第三方库,需查阅其文档是否导出了可比的哨兵错误;否则应优先考虑 errors.As

errors.As 提取底层错误值并做类型断言

errors.As 用于从错误链中向下查找第一个能被赋值给指定类型的错误值,并将其实例存入目标变量。它解决的是“这个错误底层是不是某种具体类型”的问题。

典型使用场景:你收到一个 os.PathError 包裹的错误,想取出里面的 Err 字段看是不是 syscall.EACCES;或者处理 net.OpError 时想提取底层的 syscall.Errno

立即学习go语言免费学习笔记(深入)”;

  • 目标变量必须是指针类型(如 *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)
}

为什么不能只用 == 或 switch err.(type)?

直接用 err == io.EOF 在简单场景可行,但一旦错误被 fmt.Errorf("wrap: %w", err)errors.Wrap(旧库)包裹,就失效了——因为外层错误不是 io.EOF 本身。

AIPAI
AIPAI

AI视频创作智能体

下载

switch err.(type) 只能匹配最外层错误类型,无法穿透包装。比如 fmt.Errorf("read failed: %w", os.ErrNotExist) 的类型是 *fmt.wrapError,不是 *os.PathError,更不是 os.ErrNotExist 的底层类型。

  • errors.Iserrors.As 是 Go 1.13+ 错误链标准方案,专为多层包装设计
  • 手动递归调用 errors.Unwrap 实现类似逻辑非常容易漏层或 panic,不推荐
  • 第三方错误包装库(如 github.com/pkg/errors)在 Go 1.13+ 后基本可弃用,因其 Causer / Unwrapper 接口已被标准库覆盖

嵌套错误里混用 Is 和 As 的实际顺序

真实错误往往既需要判断“是不是某个哨兵错误”,又需要提取“底层是不是某种结构体”。二者不互斥,但有逻辑先后:通常先 errors.Is 快速识别已知错误,再用 errors.As 做精细处理。

容易踩的坑:在未确认错误非 nil 的情况下直接传给 errors.As,会导致 panic(因内部对 nil 解引用)。虽然标准库文档没明说,但实测 errors.As(nil, &x) 返回 false 不 panic;不过为保险起见,仍建议显式判空。

  • 总是先检查 err != nil,再调用 errors.Iserrors.As
  • 如果 errors.Is(err, someSentinel) 为 true,通常无需再 As ——除非你要访问该哨兵错误的字段(但哨兵错误一般无字段)
  • 如果 errors.As(err, &x) 成功,可进一步对 xerrors.Is(x.Err, ...) 判断其内部错误

复杂点在于错误链可能深达 5–6 层,且中间混用标准库和旧式包装;这时候靠肉眼调试几乎不可行,建议配合 fmt.Printf("%+v", err) 查看完整链路。

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
golang如何定义变量
golang如何定义变量

golang定义变量的方法:1、声明变量并赋予初始值“var age int =值”;2、声明变量但不赋初始值“var age int”;3、使用短变量声明“age :=值”等等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

182

2024.02.23

golang有哪些数据转换方法
golang有哪些数据转换方法

golang数据转换方法:1、类型转换操作符;2、类型断言;3、字符串和数字之间的转换;4、JSON序列化和反序列化;5、使用标准库进行数据转换;6、使用第三方库进行数据转换;7、自定义数据转换函数。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

229

2024.02.23

golang常用库有哪些
golang常用库有哪些

golang常用库有:1、标准库;2、字符串处理库;3、网络库;4、加密库;5、压缩库;6、xml和json解析库;7、日期和时间库;8、数据库操作库;9、文件操作库;10、图像处理库。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

343

2024.02.23

golang和python的区别是什么
golang和python的区别是什么

golang和python的区别是:1、golang是一种编译型语言,而python是一种解释型语言;2、golang天生支持并发编程,而python对并发与并行的支持相对较弱等等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

210

2024.03.05

golang是免费的吗
golang是免费的吗

golang是免费的。golang是google开发的一种静态强类型、编译型、并发型,并具有垃圾回收功能的开源编程语言,采用bsd开源协议。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

396

2024.05.21

golang结构体相关大全
golang结构体相关大全

本专题整合了golang结构体相关大全,想了解更多内容,请阅读专题下面的文章。

240

2025.06.09

golang相关判断方法
golang相关判断方法

本专题整合了golang相关判断方法,想了解更详细的相关内容,请阅读下面的文章。

194

2025.06.10

golang数组使用方法
golang数组使用方法

本专题整合了golang数组用法,想了解更多的相关内容,请阅读专题下面的文章。

458

2025.06.17

C++ 设计模式与软件架构
C++ 设计模式与软件架构

本专题深入讲解 C++ 中的常见设计模式与架构优化,包括单例模式、工厂模式、观察者模式、策略模式、命令模式等,结合实际案例展示如何在 C++ 项目中应用这些模式提升代码可维护性与扩展性。通过案例分析,帮助开发者掌握 如何运用设计模式构建高质量的软件架构,提升系统的灵活性与可扩展性。

8

2026.01.30

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
Git 教程
Git 教程

共21课时 | 3.1万人学习

Git版本控制工具
Git版本控制工具

共8课时 | 1.5万人学习

Git中文开发手册
Git中文开发手册

共0课时 | 0人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号