前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >Golang 语言怎么使用 panic 函数?

Golang 语言怎么使用 panic 函数?

作者头像
frank.
发布2021-03-09 18:18:06
8.1K0
发布2021-03-09 18:18:06
举报

01

介绍

panic 是一个 Go 内置函数,它用来停止当前常规控制流并启动 panicking(运行时恐慌)过程。当函数 F 调用 panic 函数时,函数 F 的执行停止,函数 F 中已进行了求值的 defer 函数都将得到正常执行,然后函数 F 将控制权返还给其调用者。对于函数 F 的调用者而言,函数 F 之后的行为就如同调用者调用的函数是 panic 一样,该 panicking(运行时恐慌)过程将继续在栈上进行下去,直到当前 goroutine 中的所有函数都返回为止,此时程序将崩溃退出。

02

panic 触发方式和引发的后果

Golang 语言是静态强类型语言,在编译时,大多数问题就会被发现。但是一些会触发 panic 的问题只能在运行时才会被发现。

panic 触发方式有两种,除了上面讲到的,在运行时遇到错误触发 panic,比如越界访问数组,不相同类型的变量强制类型转换等,还可以通过直接调用 panic 函数触发 panic。

怎么通过显式调用 panic 函数触发 panic,panic 函数接收一个 interface{} 空接口类型的参数,也就是说,panic 函数可以接收一个任意类型的参数,代码如下:

代码语言:javascript
复制
func panic(v interface{})

什么时候通过显式调用 panic 函数触发 panic?

虽然 panic 可以使程序崩溃,我们尽量少用 panic,但是少用不等于不用。阅读过 golang 源码的读者应该发现在 golang 标准库代码中有显式调用 panic 函数的代码片段,比如 golang 标准库的 json 包。

请参阅在 encode.go 文件中 encodeState 类型的 error 和 marshal 方法的代码。

另外,当我们在程序中处理会影响程序正确运行的错误时,也可以考虑使用显式调用 panic 函数来返回错误。

不管是显式调用 panic 函数,还是运行时检测到违法情况自动触发 panic,都会导致程序崩溃。那么,我们应该怎么处理 panic 呢?

通常的做法是使用 defer 和 recover 捕获 panic,将 panic 错误写入日志文件,将程序恢复正常执行。需要注意的是,panic 是谁触发谁捕获,当我们调用三方库时,调用方是不会考虑处理三方库的 panic 异常。

但是,对于一些严重的 panic 异常,例如 main 函数和 init 函数中执行的程序代码,不应该使用 recover 捕获并将程序恢复正常执行,而是应该及时让 panic 执行,使程序崩溃,及时暴露出问题并解决。

03

使用 defer 和 recover 捕获 panic

defer 是什么?

defer 语句将延迟调用函数保存到列表上。defer 所在的函数返回后,将执行保存的延迟调用函数列表。

defer 延迟函数有 3 个规则:

  1. 在对 defer 语句求值时,将对 defer 延迟调用的函数的参数求值。
  2. defer 所在的函数返回后,将按照后进先出的顺序执行 defer 保存的延迟调用函数。
  3. defer 延迟调用函数可以读取并分配给返回函数的命名返回值。

注意:关于 defer 的知识点还有很多,感兴趣的读者可以查阅相关资料进一步了解。因为本文的重点是讲述 panic,所以不再对 defer 做过多的赘述。

recover 是什么?

recover 是一个 Go 内置函数,可以重新获取对一个运行时恐慌的 goroutine 的控制。recover 仅在 defer 延迟函数内部使用。在正常执行程序中,调用 recover 函数,将返回 nil。如果当前 goroutine 处于恐慌状态,调用 recover 会捕获提供给 panic 的值并恢复正常执行。

使用 defer 和 recover 捕获 panic

通常,我们不会去捕获运行时 panic,发生 panic 异常,直接让程序崩溃即可,及时根据 panic 提供的信息,修复异常。但是,一些情况下,我们还是需要捕获 panic,比如在程序发生 panic 异常时,释放资源。比如关闭文件或者释放锁。

上面我们讲到,调用 recover 函数可以捕获 panic,但是 recover 函数仅在 defer 函数内部使用。所以想要使用 recover 捕获 panic,我们需要结合 defer 一起使用。而且,程序在触发 panic 异常后,虽然不会继续往下执行代码,但是可以执行 defer 调用的函数,示例代码如下:

未使用 defer 调用匿名函数的 recover 捕获 panic:

代码语言:javascript
复制
func main() {
    fmt.Println("this is a panic example")
    panic("this is a panic")
    r := recover()
    fmt.Printf("panic recover:%s", r)
}

output:

代码语言:javascript
复制
./prog.go:10:5: unreachable code
Go vet exited.

this is a panic example
panic: this is a panic

goroutine 1 [running]:
main.main()
 /tmp/sandbox752240852/prog.go:9 +0x95

Program exited: status 2.

使用 defer 调用匿名函数的 recover 捕获 panic:

代码语言:javascript
复制
func main() {
    fmt.Println("this is a panic example")
    defer func(){
        if r := recover(); r != nil {
            fmt.Printf("panic recover:%s", r)
        }
    }()
    panic("this is a panic")
    // r := recover()
    // fmt.Printf("panic recover:%s", r)
}

output:

代码语言:javascript
复制
this is a panic example
panic recover:this is a panic
Program exited.

04

总结

本文主要介绍在 golang 语言中,panic 异常的概念,发生 panic 的两种方式和导致的后果,以及捕获 panic 的方法。因为 panic 导致的后果非常严重,会导致程序崩溃,所以我们在处理一些不会影响程序正确运行的错误时,尽量使用 error 处理错误。

推荐阅读:

Golang 语言怎么处理错误?

Go 语言学习之错误处理

Go语言学习之 panic 和 recover

Golang 语言的值验证库 Validator 怎么使用?

Go team 开源项目 Go Cloud 使用的依赖注入工具 Wire 怎么使用?

参考资料: https://blog.golang.org/defer-panic-and-recover

本文参与?腾讯云自媒体分享计划,分享自微信公众号。
原始发表:2021-03-04,如有侵权请联系?cloudcommunity@tencent.com 删除

本文分享自 Go语言开发栈 微信公众号,前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与?腾讯云自媒体分享计划? ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档
http://www.vxiaotou.com