Go 错误处理

错误

​ 在Go语言中, 错误是可以预期的,并且不是非常严重,不会影响程序的运行。对于这类问题,返回错误给调用者的方法,让调用者自行处理。

error接口

​ 在Go语言中,错误是通过内置 error 接口实现的。它很简单,只有一个Error方法来返回具体的错误信息。

type error interface {
    Error() string
}


package errors
// New returns an error that formats as the given text.
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
}

​ 一般而言,error接口用于当方法或者函数执行遇到错误时进行返回,而且是第二个值。通过这种方式,可以让调用者自己根据错误信息决定如何处理。

提示:方法和函数基本上差不多,区别在于有无接收者,所以方法和函数表达的是一个意思。

error 工厂函数

​ 我们可以自定义函数返回错误信息给调用者。通过 **errors.New()**这个工厂函数生成错误信息,它接收一个字符串参数,返回一个error接口

自定义error

​ 工厂函数返回错误信息的方式只能传递一个字符串,如果想携带更多信息需要自定义error。自定义error其实就是先定义一个新类型,比如结构体,然后让这个类型实现error接口。

type commonError struct {
    errorCode int // 错误码
    errorMsg string // 错误信息
}

func (c *commonError) Error() stirng {
    return c.errorMsg
}

return 0, &commonError{
    errorCode : 1,
    errorMSg : "不能为空",
}

error断言

​ 有了自定义error,可以携带更多的错误信息,需要把返回的error接口转换为自定义的错误类型,需要用到类型断言。类型断言在error接口上的应用,也称之为error断言。

sum, err := add(-1, 2)
if cm, ok := sum.(*commonError); ok {
    fmt.Println("错误代码为:",cm.errorCode,",错误信息为:",cm.errorMsg)
} else {
    fmt.Println(sum)
}

错误嵌套

Error Wrapping

​ error接口虽然比较简洁,但是功能相对较弱。如果基于一个存在的error再生成一个error,这就是错误嵌套。

// 自定义结构体
type MyError struct {
    err error 
    msg string
}

// 实现error接口,并在初始化MyError的时候传递存在的error和新的错误信息
func (e *MyError)Error() string {
    return e.err.Error() + e.msg
}

func test() {
    // err 是一个存在的错误
    newErr := MyError(err, "错误信息")
}

​ 上述方式可以满足需求,但是很繁琐,需要自定义新的类型还要实现error接口。所以从Go语言1.13版本开始,Go标准库新增了Error Wrapping 功能 ,让我们可以基于一个存在的error生成新的error,并且保留原error信息。

// Go语言没有提供Wrap函数,而是扩展了fmt.Errorf函数,增加一个%w,通过这种方式,可以生成wrapping error

e := errors.New("错误信息")
w := fmt.Errorf("wrap一个错误:%w", e)

errors.Unwrap函数

// error可以包括嵌套生成新error,也可以被解开,通过errors.Unwrap函数得到被嵌套的error
// Go语言提供 errors.Unwrap 用于获取被嵌套的error,获取原始错误
fmt.Println(errors.Unwrap(w))

// 输出
错误信息

errors.Is函数

​ 由于Go语言的Error Wrapping功能,让人不知道返回的err是否被嵌套,又嵌套了几层?于是Go语言提供了error.Is 函数,用来判断两个error是否是同一个

func Is(err, target error) bool
  • 如果err和target是同一个,返回true
  • 如果err是一个wrapping error,target也包含在这个嵌套error链中的话,返回true。

errors.As函数

​ 同理,error嵌套之后,error断言也不能用了,所以Go语言为解决这个问题,提供了errors.As函数。所以我们要尽可能使用Is、As这些函数做判断和转换。

var cm *commonError

if errors.As(err, &cm) {
     fmt.Println("错误代码为:",cm.errorCode,",错误信息为:",cm.errorMsg)
} else {
   fmt.Println(sum)
}

Deferred函数

​ Go语言提供了defer函数,保证不管自定义的函数出现异常还是错误,都会被执行。defer 关键字用于修饰一个函或者方法,使得该函数或者方法在返回前才会执行,也就是被延迟,但又可以保证一定会执行。

defer语句通常被用于成对操作,如文件的打开和关闭,加锁和释放锁,连接的建立和断开等。不管多复杂的操作,都可以保证资源被正确的释放。

  • 一个方法或者函数中,可以有多个defer语句
  • 多个deger语句的执行顺序依照后进先出的原则

​ defer有一个调用栈,越早定义的越靠近栈的底部,在执行defer语句的时候,会从栈顶弹出一个defer然后执行。

Panic异常

Go语言是一门静态的强类型语言,很多问题都尽可能在编译时捕获,但是有一些只能在运行时检查,比如数组越界、不同类型强制转换,这类运行时问题会引起panic异常。除了运行时可以产生panic异常,我们也可以自己抛出panic异常。比如数据库连接。

​ panic是Go语言内置的函数,可以接受interface{}类型的参数,也就是任何类型的值都可以传给panic函数。

// interface{} 是空接口的意思,在Go语言中代表任意类型
func painc(v interface{})

​ panic异常是一种非常严重的情况,会让程序终端执行,使程序崩溃,所以如果不是影响程序运行的错误,不要使用panic,使用普通error即可。

Recover捕获Panic异常

​ 通常情况下,我们不对panic异常做任何处理,已然已经影响到程序运行了,就直接崩溃即可。但也有些特殊情况,比如要在程序崩溃前释放资源,这时候需要从panic异常中恢复,才能完成处理。

​ 在Go语言中,可以通过内置的Recover函数恢复panic异常。因为在程序panic异常崩溃的时候,之后被defer修饰的函数才能被执行,所以recover函数要结合defer关键字使用才能生效

// 通过defer关键字 + 匿名函数 + recover 函数从panic异常中恢复

defer func() {
    if p := recover(); p != nil {
        fmt.Println(p)
    }
}()

// recover函数成功捕获panic异常,返回值是通过panic函数传递的参数值

上一篇:.net5 WebAPI 参数验证 ModelValid


下一篇:ThinkPHP5有关模型hasOne、hasMany、belongsTo详解