一文掌握Golang中Panic與Recover的作用和使用方法
panic
panic作用是終止當前正在運行的程序(包括所有協程)并輸出導致異常的堆棧信息。在遇到無法處理的異常情況時,例如比如數組越界、操作未初始化的map、空指針等都會觸發panic。主動觸發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
數據越界導致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可以讓觸發了panic的程序繼續運行,recover僅在延遲函數defer中有效,在正常的執行過程中,調用recover會返回nil并且不產生其他任何效果。如果當前的goroutine觸發了panic,調用recover可以捕獲到panic的輸入值,并且恢復正常運行。這個特性對于像web服務就非常有用了, 當web服務處理某個請求時,某個方法觸發了panic,這時候顯然是不應該直接讓web服務掛掉的。這種場景下,就可以使用recover來捕獲panic并且讓服務正常運行下去。
在其他語言里,通常是底層拋出異常,上層邏輯通過try/catch捕獲異常。defer/panic/recover配合使用可以實現類似try/catch的功能。
將Recover()寫在defer中,在可能發生panic的代碼之前執行defer,當程序觸發panic后,系統將跳過后面的代碼,按照逆序執行已經注冊的defer函數,如果defer函數中調用了recover(),recover()會返回捕獲到的panic的錯誤信息。
使用recover需要注意幾點:
- recover需要在defer的方法里面直接調用,不能對recover()包一層方法后再在defer的方法里面調用
- recover只能捕獲同一個協程中的panic,無法捕獲其它協程的panic
defer/panic/recover示例
成功捕獲實例一
func main() {
defer func() {
if err := recover(); err != nil {
fmt.Printf("recover:%v\n", err)
}
}()
panic("an error occurred")
}
成功捕獲實例二
func main() {
defer func() {
if err := recover(); err != nil {
fmt.Printf("recover:%v\n", err)
}
}()
test()
}
func test() {
panic("an error occurred")
}
成功捕獲實例三
func main() {
test()
}
func test() {
defer func() {
if err := recover(); err != nil {
fmt.Printf("recover:%v\n", err)
}
}()
panic("an error occurred")
}
不能捕獲實例一
func main() {
if err := recover(); err != nil {
fmt.Printf("recover:%v\n", err)
}
panic("an error occurred")
}
不能捕獲實例二
func main() {
go func() {
defer func() {
if err := recover(); err != nil {
fmt.Printf("recover:%v\n", err)
}
}()
}()
panic("an error occurred")
}
不能捕獲實例三
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")
}
不能捕獲實例四
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")
}
不能捕獲實例五
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")
}()
}
小結
本文介紹了panic和recover的作用及使用方法,以及defer/panic/recover配合使用實現類似try/catch的功能,下篇文章將從源碼角度來做講解。