go - 使用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!: <nil> --- is traceable: true
//pointer wrapper error trace --- whoa!: Hi!: <nil> --- is traceable: false
https://play.golang.org/p/-hSukZ-gii2
似乎参数的任何差异,包括包装的错误参数,Err
都会导致无法找到类型errors.Is()
。
解决方案
false
您在尝试在另一个通过中查找一个错误时得到的原因errors.Is
是因为 - 虽然这两个错误可能具有相同的字段值 - 它们是两个不同的内存指针:
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)}
}
推荐阅读
- javascript - Asp Core 5.0 Razor Page:调用 RedirectToPage() 时如何保留 window.location.hash
- python - 正确绘制混淆矩阵
- c# - 找不到类型或命名空间名称 TSocket apache thrift C# Visual Studio 2019
- typescript - 为什么这些在打字稿中的处理方式不同
- vba - 什么是“编译错误无法在对象模块中定义公共用户定义类型”?: VBA
- android - 如何在没有看到取消选中操作的情况下显示未选中的 RadioButton?
- angular - 动态创建多个农业网格表
- linux - 用“直接”符号链接替换多级符号链接
- node.js - 如何访问平面图的不同资源并保存每个资源在数据库中的位置
- excel - ThisWorkbook.ActiveSheet.PageSetup.PrintArea 返回值“”