一文掌握Golang中Panic與Recover的作用和使用方法
panic
panic作用是終止當(dāng)前正在運(yùn)行的程序(包括所有協(xié)程)并輸出導(dǎo)致異常的堆棧信息。在遇到無法處理的異常情況時(shí),例如比如數(shù)組越界、操作未初始化的map、空指針等都會觸發(fā)panic。主動觸發(fā)panic示例:
package main
func main() {
// 未處理的自定義異常
customException := "an error occurred"
panic(customException)
}
會輸出如下信息:
panic: an error occurred
goroutine 1 [running]:
main.main()
/Users/ning/projects/go/workspace/hello/panic/main.go:7 +0x34
Process finished with the exit code 2
數(shù)據(jù)越界導(dǎo)致panic示例:
package main
import "fmt"
func main() {
a := [2]int{4, 5}
fmt.Println(a[3])
}
會輸出如下信息:
# command-line-arguments
./main.go:11:16: invalid argument: array index 3 out of bounds [0:2]
Compilation finished with exit code 2
recover
recover可以讓觸發(fā)了panic的程序繼續(xù)運(yùn)行,recover僅在延遲函數(shù)defer中有效,在正常的執(zhí)行過程中,調(diào)用recover會返回nil并且不產(chǎn)生其他任何效果。如果當(dāng)前的goroutine觸發(fā)了panic,調(diào)用recover可以捕獲到panic的輸入值,并且恢復(fù)正常運(yùn)行。這個(gè)特性對于像web服務(wù)就非常有用了, 當(dāng)web服務(wù)處理某個(gè)請求時(shí),某個(gè)方法觸發(fā)了panic,這時(shí)候顯然是不應(yīng)該直接讓web服務(wù)掛掉的。這種場景下,就可以使用recover來捕獲panic并且讓服務(wù)正常運(yùn)行下去。
在其他語言里,通常是底層拋出異常,上層邏輯通過try/catch捕獲異常。defer/panic/recover配合使用可以實(shí)現(xiàn)類似try/catch的功能。
將Recover()寫在defer中,在可能發(fā)生panic的代碼之前執(zhí)行defer,當(dāng)程序觸發(fā)panic后,系統(tǒng)將跳過后面的代碼,按照逆序執(zhí)行已經(jīng)注冊的defer函數(shù),如果defer函數(shù)中調(diào)用了recover(),recover()會返回捕獲到的panic的錯(cuò)誤信息。
使用recover需要注意幾點(diǎn):
- recover需要在defer的方法里面直接調(diào)用,不能對recover()包一層方法后再在defer的方法里面調(diào)用
- recover只能捕獲同一個(gè)協(xié)程中的panic,無法捕獲其它協(xié)程的panic
defer/panic/recover示例
成功捕獲實(shí)例一
func main() {
defer func() {
if err := recover(); err != nil {
fmt.Printf("recover:%v\n", err)
}
}()
panic("an error occurred")
}
成功捕獲實(shí)例二
func main() {
defer func() {
if err := recover(); err != nil {
fmt.Printf("recover:%v\n", err)
}
}()
test()
}
func test() {
panic("an error occurred")
}
成功捕獲實(shí)例三
func main() {
test()
}
func test() {
defer func() {
if err := recover(); err != nil {
fmt.Printf("recover:%v\n", err)
}
}()
panic("an error occurred")
}
不能捕獲實(shí)例一
func main() {
if err := recover(); err != nil {
fmt.Printf("recover:%v\n", err)
}
panic("an error occurred")
}
不能捕獲實(shí)例二
func main() {
go func() {
defer func() {
if err := recover(); err != nil {
fmt.Printf("recover:%v\n", err)
}
}()
}()
panic("an error occurred")
}
不能捕獲實(shí)例三
func main() {
defer func() {
if err := recover(); err != nil {
fmt.Printf("recover:%v\n", err)
}
}()
go test()
for {
select {}
}
}
func test() {
panic("an error occurred")
}
不能捕獲實(shí)例四
func main() {
defer func() {
recoverFromPanic()
}()
test()
}
func recoverFromPanic() {
if err := recover(); err != nil {
fmt.Printf("recover:%v\n", err)
}
}
func test() {
panic("an error occurred")
}
不能捕獲實(shí)例五
func main() {
defer func() {
if err := recover(); err != nil {
fmt.Printf("recover:%v\n", err)
}
}()
test()
for {
select {}
}
}
func test() {
go func() {
panic("an error occurred")
}()
}
小結(jié)
本文介紹了panic和recover的作用及使用方法,以及defer/panic/recover配合使用實(shí)現(xiàn)類似try/catch的功能,下篇文章將從源碼角度來做講解。