Go语言中错误处理的方式总结

 更新时间:2023年07月05日 10:35:54   作者:242030  
这篇文章会结合 errors 中的函数,来讨论一下 Go 中常见的 error 使用方式,文中的示例代码讲解详细,具有一定的学习价值,需要的可以了解一下

Go 的 error 有两个很重要的特性:

  • error 就是一个普通的值,处理起来没有额外的开销
  • error 的扩展性很不错,可以按照不同的场景来自定义错误

在 Go1.13 之后,在 errors 包中提供了一些函数,让错误的处理和追踪更加方便一些。

这篇文章会结合 errors 中的函数,来讨论一下 Go 中常见的 error 使用方式。

这里说的 errors 包是指 Go 中的原生 errors 包。

1、原生 error

在 Go 的错误处理中,下面的代码占绝大多数:

if err != nil {
   return err
}

在满足业务需求的情况下,这种错误处理其实是最推荐的方式,这种直接透传的方式让代码之间的耦合度更低。在很多情况下,如果不关心错误中的具体信息,使用这种方式就可以了。

2、提前定义好 error

原生的 error 在有些情况下使用起来就不是很方便,比如我需要获得具体的错误信息,如果还用上面的方式来使用error,可能会出现下面的代码:

if err != nil && err.Error() == "invalid param" {
}

写过代码的都知道上面的代码很不优雅,另外如果错误的信息变化之后,这里的代码逻辑就会出错,可以通过把错误定义成一个变量:

var (
    ErrInvalidParam = errors.New("invalid param")
)

那么上面的代码就可以变成这样:

if err != nil && err == ErrInvalidParam {
}

如果一次性需要处理的错误比较多,还可以使用 switch 进行处理:

if err != nil {
    switch err {
    case ErrInvalidParam:
        return
    case ErrNetWork:
        return
    case ErrFileNotExist:
        return
    default:
        return
    }
}

但是这种方式还不完美,因为 error 在传递的过程中,有可能会被包装,以携带更多的堆栈信息,比如下面这样:

if err != nil {
    // 在包装错误的时候,这里格式化错误要使用%w
    return fmt.Errorf("add error info: %+v, origin error: %w", "other info", err)
}

假设上面被包装的错误是 ErrInvalidParam,那么在调用的地方判断错误,就不能使用下面的代码:

if err != nil && err == ErrInvalidParam {
}

为了解决这个问题, errors.Is 函数可以判断被包装的 error 中是否有预期的 error:

if errors.Is(err, ErrInvalidParam) {
}

尽量使用 errors.Is 来替代对 error 的比较。

3、使用自定义的错误类型

上面的 error 使用方式在某些情况下还是不能满足要求。假如对于上面的无效参数 error,业务方想要知道具体是哪个参数无效,直接定义的错误就无法满足要求。error 本质是一个接口,也就是是说,只要实现了 Error 方法,就是一个 error 类型:

type error interface {
    Error() string
}

那么就可以自定义一种错误类型:

type ErrInvalidParam struct {
    ParamName  string
    ParamValue string
}
func (e *ErrInvalidParam) Error() string {
    return fmt.Sprintf("invalid param: %+v, value: %+v", e.ParamName, e.ParamValue)
}

然后就可以使用类型断言机制或者类型选择机制,来对不同类型的错误进行处理:

e, ok := err.(*ErrInvalidParam)
if ok && e != nil {
}

同样可以在 switch 中使用:

if err != nil {
    switch err.(type) {
    case *ErrInvalidParam:
        return
    default:
        return
    }
}

在这里 error 同样会存在被包装的问题,而 errors.As 刚好可以用来解决这个问题,可以判断出被包装的错误中是否存在某个 error 类型:

var e *ErrInvalidParam
if errors.As(err, &e) {
}

4、更灵活的 error 类型

上面的方式已经可以解决大部分场景的 error 处理了,但是在一些复杂的情况下,可能需要从错误中获取更多的信息,还包含一定的逻辑处理。

在 Go 的 net 包中,有这样的一个接口:

type Error interface {
    error
    Timeout() bool  
    Temporary() bool
}

在这个接口中,有两个方法,这两个方法会对这个错误类型进行处理,判断是超时错误还是临时错误,实现了这个接口的 error 要实现这两个 方法,实现具体的判断逻辑。

在处理具体 error 时,会调用相应的方法来判断:

if ne, ok := e.(net.Error); ok && ne.Temporary() { 
     // 对临时错误进行处理 
}
if ne, ok := e.(net.Error); ok && ne.Timeout() { 
     // 对超时错误进行处理 
}

这种类型的 error 相对来说,使用的会比较少,一般情况下,尽量不要使用这么复杂的处理方式。

5、errors 中的其他能力

在 errors 包中,除了上面提到的 errors.Is 和 errors.As 两个很有用的函数之外,还有一个比较实用的函数errors.Unwrap,这个函数可以从包装的错误中将原错误解析出来。

可以使用 fmt.Errorf 来包装 error,需要使用 %w 的格式化:

return fmt.Errorf("add error info: %+v, origin error: %w", "other info", err)

在后续的 error 处理时,可以调用 errors.Unwrap 函数来获得被包装前的 error:

err = errors.Unwrap(err)
fmt.Printf("origin error: %+v\n", err)
package main
import (
	"errors"
	"fmt"
)
func main(){
	err1 := errors.New("zero")
	fmt.Printf("origin error: %+v\n", err1)
	err2 := fmt.Errorf("add error info: %+v, origin error: %w", "other info", err1)
	fmt.Printf("origin error: %+v\n", err2)
	err3 := errors.Unwrap(err2)
	fmt.Printf("origin error: %+v\n", err3)
}

程序输出

origin error: zero
origin error: add error info: other info, origin error: zero
origin error: zero

6、注意

如果需要使用 goroutine 时,应该使用统一的 Go 函数进行创建,这个函数中会进行 recover ,避免因为野生goroutine panic 导致主进程退出。

func Go(f func()){
    go func(){
        defer func(){
            if err := recover(); err != nil {
                log.Printf("panic: %+v", err)
            }
        }()
        f()
    }()
}

在应用程序中出现错误时,使用 errors.New 或者 errors.Errorf 返回错误:

errors.Errorf("用户余额不足, uid: %d, money: %d", uid, money)

如果是调用应用程序的其他函数出现错误,请直接返回,如果需要携带信息,请使用 errors.WithMessage。

// https://github.com/pkg/errors
errors.WithMessage(err, "其他附加信息")

如果是调用其他库(标准库、企业公共库、开源第三方库等)获取到错误时,请使用 errors.Wrap 添加堆栈信息。

切记,不要每个地方都是用 errors.Wrap 只需要在错误第一次出现时进行 errors.Wrap 即可。

根据场景进行判断是否需要将其他库的原始错误吞掉,例如可以把 repository 层的数据库相关错误吞掉,返回业务错误码,避免后续我们分割微服务或者更换 ORM 库时需要去修改上层代码。

注意我们在基础库,在大量引入的第三方库编写时一般不使用 errors.Wrap 避免堆栈信息重复。

func f() error {
    err := json.Unmashal(&a, data)
    if err != nil {
        return errors.Wrap(err, "其他附加信息")
    }
    // 其他逻辑
    return nil
}

禁止每个出错的地方都打日志,只需要在进程的最开始的地方使用 %+v 进行统一打印,例如 http/rpc 服务的中间件。

错误判断使用 errors.Is 进行比较:

func f() error {
    err := A()
    if errors.Is(err, io.EOF){
        return nil
    }
    // 其他逻辑
    return nil
}

错误类型判断,使用 errors.As 进行赋值:

func f() error {
    err := A()
    var errA errorA
    if errors.As(err, &errA){
    }
    // 其他逻辑
    return nil
}

以上就是Go语言中错误处理的方式总结的详细内容,更多关于Go错误处理的资料请关注脚本之家其它相关文章!

相关文章

  • 利用Go语言搭建WebSocket服务端方法示例

    利用Go语言搭建WebSocket服务端方法示例

    这篇文章主要给大家介绍了利用Go语言搭建WebSocket服务端方法,文中通过示例代码介绍的非常详细,需要的朋友们可以参考借鉴,下面来一起看看吧。
    2017-04-04
  • 使用Golang进行比较版本号大小

    使用Golang进行比较版本号大小

    在日常开发中,比较版本号大小的情况是经常遇到的,这篇文章主要为大家详细介绍了如何使用Golang进行比较版本号大小,需要的小伙伴可以参考下
    2024-01-01
  • 使用Golang的channel交叉打印两个数组的操作

    使用Golang的channel交叉打印两个数组的操作

    这篇文章主要介绍了使用Golang的channel交叉打印两个数组的操作,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧
    2021-04-04
  • 详解Golang中的交叉编译

    详解Golang中的交叉编译

    在 Golang 中,交叉编译指的是在同一台机器上生成针对不同操作系统或硬件架构的二进制文件,这在开发跨平台应用或构建特定平台的发布版本时非常有用,本文就详细的给大家介绍一下Golang中的交叉编译,需要的朋友可以参考下
    2023-08-08
  • 浅析Go语言版本的forgery

    浅析Go语言版本的forgery

    使用过Python语言的朋友们可能使用过 forgery_py ,它是一个伪造数据的工具。这篇文章主要介绍了Go语言版本的forgery,需要的朋友可以参考下
    2018-08-08
  • 使用Go语言实现谷歌翻译功能

    使用Go语言实现谷歌翻译功能

    这篇文章主要为大家详细介绍了如何使用Go语言实现谷歌翻译功能,文中的示例代码讲解详细,具有一定的借鉴价值,有需要的小伙伴可以参考下
    2024-02-02
  • Go语言-为什么返回值为接口类型,却返回结构体

    Go语言-为什么返回值为接口类型,却返回结构体

    这篇文章主要介绍了Go语言返回值为接口类型,却返回结构体的实例讲解,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧
    2021-04-04
  • Go modules replace解决Go依赖引用问题

    Go modules replace解决Go依赖引用问题

    这篇文章主要为大家介绍了Go modules replace解决Go依赖引用问题,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
    2022-06-06
  • golang中的单引号转义问题

    golang中的单引号转义问题

    这篇文章主要介绍了golang中的单引号转义问题,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2023-02-02
  • 深入了解Golang 哈希算法之MD5、SHA-1和SHA-256

    深入了解Golang 哈希算法之MD5、SHA-1和SHA-256

    哈希算法是计算机科学领域中一种重要的技术,它将任意长度的输入数据映射为固定长度的哈希值,在本篇文章中,我们将深入探讨Golang中的哈希算法,从多个方面介绍其详细内容,希望通过本文的阅读你将对 Golang哈希算法有更全面的理解
    2023-05-05

最新评论