SentinelError模式:错误变量定义技巧
时间:2026-02-13 13:30:40 404浏览 收藏
小伙伴们有没有觉得学习Golang很有意思?有意思就对了!今天就给大家带来《Sentinel Error模式:定义预定义错误变量的最佳实践》,以下内容将会涉及到,若是在学习中对其中部分知识点有疑问,或许看了本文就能帮到你!
应该,但需封装:用 var 声明顶层哨兵错误,优先 errors.New;需上下文或扩展时升级为自定义类型并实现 Unwrap() 返回自身;包装必须用 %w,导出错误名以 Err 开头且慎用。

Go 里该不该用 var ErrXXX = errors.New("xxx") 定义 Sentinel 错误?
应该,但得加一层封装。直接裸用 errors.New 创建的错误变量,无法携带上下文、不支持比较语义(比如 errors.Is)、也不利于后期扩展为带字段的错误类型。
常见错误现象:写了个 var ErrNotFound = errors.New("not found"),后来想在日志里加请求 ID,发现没法动态注入;或者调用方用 == 比较失败——因为 errors.New 每次都新建实例,地址不同。
- 始终用
var声明顶层错误变量,别在函数里重复errors.New - 优先用
errors.New而不是fmt.Errorf初始化 Sentinel 错误(后者隐含格式化开销,且易误加参数) - 如果未来可能需要携带字段(如码、追踪 ID),现在就定义为自定义类型,哪怕暂时只实现
Error()方法
为什么 errors.Is 找不到你定义的 ErrXXX?
因为 errors.Is 依赖错误链中的“底层哨兵值”匹配,而你可能无意中把它包掉了。比如用了 fmt.Errorf("wrap: %w", ErrNotFound) 是 OK 的,但 fmt.Errorf("wrap: %v", ErrNotFound) 就断链了——%v 触发 Error() 输出字符串,丢掉原始错误引用。
使用场景:HTTP handler 中判断是否返回 404,或重试逻辑里跳过特定错误。
- 所有包装必须用
%w动词,不能用%s、%v或字符串拼接 - 自定义错误类型若要支持
errors.Is,需在Unwrap()方法中返回哨兵变量(或 nil) - 测试时用
errors.Is(err, pkg.ErrNotFound),别用err == pkg.ErrNotFound(后者只对未包装的顶层错误有效)
Sentinel 错误该放在 package 还是 internal/?
公开暴露的错误(调用方需要显式判断并处理的)必须放在 package 顶层;仅内部使用的哨兵错误应移入 internal/ 或私有变量(首字母小写),否则会污染导出 API。
性能影响:无。但兼容性影响大——一旦导出,就不能删、不能改类型、甚至不能改错误文本(有些下游用 strings.Contains(err.Error(), "timeout") 这种反模式硬匹配)。
- 导出错误变量名必须以
Err开头,如ErrTimeout,符合 Go 社区惯例 - 如果错误只在本包内判断(比如某个 retry loop),定义为小写变量
errInvalidState即可 - 不要为每个 HTTP 状态码都导出一个
ErrStatus400——状态码映射应由 handler 层统一处理,错误类型聚焦业务语义
要不要给 Sentinel 错误加错误码字段?
要,但别一开始就加结构体。先用纯 var + errors.New,等真出现多语言提示、监控分类、或需要和外部系统对齐码值时,再升级为带字段的类型。过早抽象反而增加调用方负担。
容易踩的坑:有人一上来就定义 type NotFoundError struct{ Code int; Msg string },结果发现 errors.Is(err, pkg.ErrNotFound) 失败——因为新类型没实现 Unwrap() 返回哨兵值。
- 升级路径:从
var ErrNotFound = errors.New("not found")→ 改为var ErrNotFound = ¬FoundError{},其中notFoundError是未导出类型 Unwrap()方法必须返回ErrNotFound自身(或 nil),才能维持errors.Is链路- 错误码字段建议用常量定义(如
const CodeNotFound = 404),别硬编码在结构体里
本篇关于《SentinelError模式:错误变量定义技巧》的介绍就到此结束啦,但是学无止境,想要了解学习更多关于Golang的相关知识,请关注golang学习网公众号!
-
505 收藏
-
503 收藏
-
502 收藏
-
502 收藏
-
502 收藏
-
179 收藏
-
290 收藏
-
223 收藏
-
351 收藏
-
151 收藏
-
159 收藏
-
428 收藏
-
280 收藏
-
446 收藏
-
325 收藏
-
104 收藏
-
196 收藏
-
- 前端进阶之JavaScript设计模式
- 设计模式是开发人员在软件开发过程中面临一般问题时的解决方案,代表了最佳的实践。本课程的主打内容包括JS常见设计模式以及具体应用场景,打造一站式知识长龙服务,适合有JS基础的同学学习。
- 立即学习 543次学习
-
- GO语言核心编程课程
- 本课程采用真实案例,全面具体可落地,从理论到实践,一步一步将GO核心编程技术、编程思想、底层实现融会贯通,使学习者贴近时代脉搏,做IT互联网时代的弄潮儿。
- 立即学习 516次学习
-
- 简单聊聊mysql8与网络通信
- 如有问题加微信:Le-studyg;在课程中,我们将首先介绍MySQL8的新特性,包括性能优化、安全增强、新数据类型等,帮助学生快速熟悉MySQL8的最新功能。接着,我们将深入解析MySQL的网络通信机制,包括协议、连接管理、数据传输等,让
- 立即学习 500次学习
-
- JavaScript正则表达式基础与实战
- 在任何一门编程语言中,正则表达式,都是一项重要的知识,它提供了高效的字符串匹配与捕获机制,可以极大的简化程序设计。
- 立即学习 487次学习
-
- 从零制作响应式网站—Grid布局
- 本系列教程将展示从零制作一个假想的网络科技公司官网,分为导航,轮播,关于我们,成功案例,服务流程,团队介绍,数据部分,公司动态,底部信息等内容区块。网站整体采用CSSGrid布局,支持响应式,有流畅过渡和展现动画。
- 立即学习 485次学习