golang Error Wrapping

go1.13中引入err wraping机制
1.13前函数返回一个error errors.New()创建一个errorString
errorString 只包含了一个字符串 实现了error接口的Error方法
当函数嵌套返回时 嵌套层数更深的error只保留了字符串信息 丢失了原始的类型

// Each call to New returns a distinct error value even if the text is identical.
func New(text string) error {
    return &errorString{text}
}

// errorString is a trivial implementation of error.
type errorString struct {
    s string
}

func (e *errorString) Error() string {
    return e.s
}
// test
func f1() error {
    return errors.New("e1")
}

func f2() error {
    e1 := f1()
    return fmt.Errorf("e2 %s", e1)
}

func main() {
    fmt.Println(f2())
}

>>> e2 e1

err wraping %w Unwrap Is As

    func main() {
    err := errors.New("raw err")
        // fmt.Errorf中的%w占位符可以wrap一个err 可以重复wrap
    errw := fmt.Errorf("wrapped err %w", err)
        // 用于比较一个err是否是target类型的error 
    fmt.Println(errors.Is(errw, err))
        // 取出被wrap的error 可以重复调用取出多层
    fmt.Println(errors.Unwrap(errw))
}

As 函数可以用来断言一个error的类型

  func main() {
    var err error
    err = &os.PathError{
        Path: "err path",
        Op:   "err op",
        Err:  errors.New("path err"),
    }
        // 无法确定wrappedErr的类型
    wrappedErr := fmt.Errorf("wrappedErr %w", err)
    if _, ok := wrappedErr.(*os.PathError); ok {
        fmt.Println("assert wrappedErr  os.PathError")
    }
        // 用As来断言wrappedErr的类型
    var perr *os.PathError
    if errors.As(wrappedErr, &perr) {
        fmt.Println("as wrappedErr os.PathError")
    }
}
>>> as wrappedErr  os.PathError

你可能感兴趣的:(golang Error Wrapping)