引
这次 errors 包算重量级更新。很有更能把以前的一些设计模式给推到。下面聊下用法。
error 装包
以前返回一个错误,想要保存 error 链,还要定义结构体保存以前的 error 信息。感兴趣看下 syscal.ECONNREFUSED 如何封装到 url.Error 的。现在只要%w 就行
err = fmt.Errorf("第二层错误信息 %w", err)
error 解包
如果是 fmt.Errorf("%w", err) 定义的错误链可以通过,或者实现了Unwrap接口的错误
e = errors.Unwrap(e) //解包错误
装包解包需要注意的地方
装包和解包错误是一一对应的,一次 Unwrap 调用,解一次 fmt.Errorf("%w", err)调用
package main
import (
"errors"
"fmt"
)
func main() {
var e, first error
first = errors.New("head")
e = fmt.Errorf("第一层错误:%w", first)
e = fmt.Errorf("第二层错误:%w", e)
e = fmt.Errorf("第三层错误:%w", e)
e = fmt.Errorf("第四层错误:%w", e)
//解包错误
e = errors.Unwrap(e)
fmt.Printf("%s\n", e)
//解包错误
e = errors.Unwrap(e)
fmt.Printf("%s\n", e)
//解包错误
e = errors.Unwrap(e)
fmt.Printf("%s\n", e)
//解包错误
e = errors.Unwrap(e)
fmt.Printf("%s\n", e)
}
/*
第三层错误:第二层错误:第一层错误:head
第二层错误:第一层错误:head
第一层错误:head
head
*/
判断错误是否相等
过去写法 ==
if err == os.ErrNotExistF {
}
现在写法 errors.Is
以前只有一个错误,现在是错误链表,要通过 errors.Is 遍历判断
b := errors.Is(err, os.ErrNotExist)
fmt.Printf("%t\n", b)
新写法根据具体类型判断 errors.As
errors.As和errors.Is的区别,主要是取出这种类型的错误
if _, err := os.Open("non-existing"); err != nil {
var pathError *os.PathError
if errors.As(err, &pathError) {
fmt.Println("Failed at path:", pathError.Path)
} else {
fmt.Println(err)
}
}
交流学习
errors 包的一些姿势还要等长时间使用才能完全开发出来,欢迎 朋友提出想法,一起学习。
github
https://github.com/guonaihong/gout
有疑问加站长微信联系(非本文作者)