Golang 语言怎么使用 panic 函数?

简介: Golang 语言怎么使用 panic 函数?

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 函数可以接收一个任意类型的参数,代码如下:

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:

func main() {
    fmt.Println("this is a panic example")
    panic("this is a panic")
    r := recover()
    fmt.Printf("panic recover:%s", r)
}

output:

./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:

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:

this is a panic example
panic recover:this is a panic
Program exited.

04

总结

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

推荐阅读:

Golang 语言怎么处理错误?

Go 语言学习之错误处理

参考资料:

https://blog.golang.org/defer-panic-and-recover


目录
相关文章
|
4月前
|
Go
Golang语言之管道channel快速入门篇
这篇文章是关于Go语言中管道(channel)的快速入门教程,涵盖了管道的基本使用、有缓冲和无缓冲管道的区别、管道的关闭、遍历、协程和管道的协同工作、单向通道的使用以及select多路复用的详细案例和解释。
147 4
Golang语言之管道channel快速入门篇
|
4月前
|
Go
Golang语言文件操作快速入门篇
这篇文章是关于Go语言文件操作快速入门的教程,涵盖了文件的读取、写入、复制操作以及使用标准库中的ioutil、bufio、os等包进行文件操作的详细案例。
73 4
Golang语言文件操作快速入门篇
|
4月前
|
Go
Golang语言之gRPC程序设计示例
这篇文章是关于Golang语言使用gRPC进行程序设计的详细教程,涵盖了RPC协议的介绍、gRPC环境的搭建、Protocol Buffers的使用、gRPC服务的编写和通信示例。
120 3
Golang语言之gRPC程序设计示例
|
4月前
|
安全 Go
Golang语言goroutine协程并发安全及锁机制
这篇文章是关于Go语言中多协程操作同一数据问题、互斥锁Mutex和读写互斥锁RWMutex的详细介绍及使用案例,涵盖了如何使用这些同步原语来解决并发访问共享资源时的数据安全问题。
101 4
|
4月前
|
Go 调度
Golang语言goroutine协程篇
这篇文章是关于Go语言goroutine协程的详细教程,涵盖了并发编程的常见术语、goroutine的创建和调度、使用sync.WaitGroup控制协程退出以及如何通过GOMAXPROCS设置程序并发时占用的CPU逻辑核心数。
84 4
Golang语言goroutine协程篇
|
4月前
|
Prometheus Cloud Native Go
Golang语言之Prometheus的日志模块使用案例
这篇文章是关于如何在Golang语言项目中使用Prometheus的日志模块的案例,包括源代码编写、编译和测试步骤。
83 3
Golang语言之Prometheus的日志模块使用案例
|
3月前
|
前端开发 中间件 Go
实践Golang语言N层应用架构
【10月更文挑战第2天】本文介绍了如何在Go语言中使用Gin框架实现N层体系结构,借鉴了J2EE平台的多层分布式应用程序模型。文章首先概述了N层体系结构的基本概念,接着详细列出了Go语言中对应的构件名称,包括前端框架(如Vue.js、React)、Gin的处理函数和中间件、依赖注入和配置管理、会话管理和ORM库(如gorm或ent)。最后,提供了具体的代码示例,展示了如何实现HTTP请求处理、会话管理和数据库操作。
49 0
|
4月前
|
Go
Golang语言错误处理机制
这篇文章是关于Golang语言错误处理机制的教程,介绍了使用defer结合recover捕获错误、基于errors.New自定义错误以及使用panic抛出自定义错误的方法。
55 3
|
4月前
|
Go
Golang语言之函数(func)进阶篇
这篇文章是关于Golang语言中函数高级用法的教程,涵盖了初始化函数、匿名函数、闭包函数、高阶函数、defer关键字以及系统函数的使用和案例。
82 3
Golang语言之函数(func)进阶篇
|
4月前
|
Go
Golang语言之函数(func)基础篇
这篇文章深入讲解了Golang语言中函数的定义和使用,包括函数的引入原因、使用细节、定义语法,并通过多个案例展示了如何定义不返回任何参数、返回一个或多个参数、返回值命名、可变参数的函数,同时探讨了函数默认值传递、指针传递、函数作为变量和参数、自定义数据类型以及返回值为切片类型的函数。
98 2
Golang语言之函数(func)基础篇