正确使用errors.Is()来进行错误类型检查
来源:stackoverflow
时间:2024-03-23 21:45:32 147浏览 收藏
本文探讨了在 Go 语言中使用 `errors.Is()` 进行错误类型检查时遇到的问题。当使用 `errors.Is()` 检查错误时,只有没有参数或带有值接收器的错误实现才能被找到。这意味着可包装的错误实现必须具有值接收器才能通过 `errors.Is()` 被找到。本文提供了使用 `errors.as` 代替 `errors.Is` 来检测和获取此类错误值的方法,并讨论了 `errors.as` 和 `errors.Is` 在指针处理方面的区别。
我正在检查 go v1.13 go v1.14 中的错误跟踪。为什么使用 errors.is()
只能找到不带参数或带值接收器的错误实现?这意味着能够包装的错误实现必须有一个值接收器,以便能够通过 errors.is()
找到。
package main import ( "fmt" "errors" ) type someAtomicError struct {} func (e *someAtomicError) Error() string { return "Hi!" } func checkAtomicError() { e := &someAtomicError{} e2 := fmt.Errorf("whoa!: %w", e) e2IsE := errors.Is(e2, &someAtomicError{}) fmt.Println("atomic error trace ---\t\t", e2, "\t\t--- is traceable: ", e2IsE) } type someWrapperError struct { Msg string Err error } func (e someWrapperError) Error() string { return fmt.Sprintf("%s: %v", e.Msg, e.Err) } func (e someWrapperError) Unwrap() error { return e.Err } func checkWrapperError() { e := someWrapperError{"Hi!", nil} e2 := fmt.Errorf("whoa!: %w", e) e2IsE := errors.Is(e2, someWrapperError{"Hi!", nil}) fmt.Println("wrapper error trace ---\t\t", e2, "\t--- is traceable: ", e2IsE) } type somePointerWrapperError struct { Msg string Err error } func (e *somePointerWrapperError) Error() string { return fmt.Sprintf("%s: %v", e.Msg, e.Err) } func (e *somePointerWrapperError) Unwrap() error { return e.Err } func checkPointerWrapperError() { e := &somePointerWrapperError{"Hi!", nil} e2 := fmt.Errorf("whoa!: %w", e) e2IsE := errors.Is(e2, &somePointerWrapperError{"Hi!", nil}) fmt.Println("pointer wrapper error trace ---\t", e2, "\t--- is traceable: ", e2IsE) } func main() { checkAtomicError() checkWrapperError() checkPointerWrapperError() } //atomic error trace --- whoa!: Hi! --- is traceable: true //wrapper error trace --- whoa!: Hi!:--- is traceable: true //pointer wrapper error trace --- whoa!: Hi!: --- is traceable: false
https://play.golang.org/p/-hsukz-gii2
参数中的任何差异(包括包装的错误参数 err
)似乎都会导致无法使用 errors.is()
找到该类型。
解决方案
当您尝试通过 errors.is
查找另一个错误时出现 false
的原因是,虽然这两个错误可能具有相同的字段值,但它们是两个不同的内存指针:
e := &somepointerwrappererror{"hi!", nil} e2 := &somepointerwrappererror{"hi!", nil} // e2 != e ew := fmt.errorf("whoa!: %w", e) errors.is(ew, e) // true errors.is(ew, e2) // false - because `ew` wraps `e` not `e2`
那么如何检测这种“类型”的错误并获取其值:使用 errors.as
代替:
e := &somepointerwrappererror{"hi!", nil} e2 := fmt.errorf("whoa!: %w", e) var ev *somepointerwrappererror if errors.as(e2, &ev) { fmt.printf("%#v\n", ev) // &somepointerwrappererror{msg:"hi!", err:error(nil)} }
https://play.golang.org/p/CttKThLasXD
远程相关,但也许它对某人有帮助:我花了一些时间才意识到 errors.as(...)
实际上需要一个指向目标的双指针,而 errors.is (...)
不会:
var _ error = (*CustomError)(nil) // ensure CustomError implements error type CustomError struct { msg string } func (e CustomError) Error() string { return e.msg } func main() { err := &CustomError{"Hello, world!"} // Methods return pointers to errors, allowing them to be nil var eval *CustomError as := errors.As(err, &eval) // yes, that's **CustomError asFaulty := errors.As(err, eval) // no compile error, so it wrongly seems okay is := errors.Is(err, eval) // that's just *CustomError fmt.Printf("as: %t, asFaulty: %t, is: %t", as, asFaulty, is) // as: true, asFaulty: false, is: true }
好了,本文到此结束,带大家了解了《正确使用errors.Is()来进行错误类型检查》,希望本文对你有所帮助!关注golang学习网公众号,给大家分享更多Golang知识!
-
502 收藏
-
502 收藏
-
501 收藏
-
501 收藏
-
501 收藏
-
139 收藏
-
204 收藏
-
325 收藏
-
477 收藏
-
486 收藏
-
439 收藏
-
357 收藏
-
352 收藏
-
101 收藏
-
440 收藏
-
212 收藏
-
143 收藏
-
- 前端进阶之JavaScript设计模式
- 设计模式是开发人员在软件开发过程中面临一般问题时的解决方案,代表了最佳的实践。本课程的主打内容包括JS常见设计模式以及具体应用场景,打造一站式知识长龙服务,适合有JS基础的同学学习。
- 立即学习 542次学习
-
- GO语言核心编程课程
- 本课程采用真实案例,全面具体可落地,从理论到实践,一步一步将GO核心编程技术、编程思想、底层实现融会贯通,使学习者贴近时代脉搏,做IT互联网时代的弄潮儿。
- 立即学习 508次学习
-
- 简单聊聊mysql8与网络通信
- 如有问题加微信:Le-studyg;在课程中,我们将首先介绍MySQL8的新特性,包括性能优化、安全增强、新数据类型等,帮助学生快速熟悉MySQL8的最新功能。接着,我们将深入解析MySQL的网络通信机制,包括协议、连接管理、数据传输等,让
- 立即学习 497次学习
-
- JavaScript正则表达式基础与实战
- 在任何一门编程语言中,正则表达式,都是一项重要的知识,它提供了高效的字符串匹配与捕获机制,可以极大的简化程序设计。
- 立即学习 487次学习
-
- 从零制作响应式网站—Grid布局
- 本系列教程将展示从零制作一个假想的网络科技公司官网,分为导航,轮播,关于我们,成功案例,服务流程,团队介绍,数据部分,公司动态,底部信息等内容区块。网站整体采用CSSGrid布局,支持响应式,有流畅过渡和展现动画。
- 立即学习 484次学习