Go中的 panic / recover 簡介與實踐記錄
簡介
go語言追求簡潔,所以go語言中沒有try…catch語句。因為go語言的作者認為將異常和控制語句混在一起,很容易讓這個程序變得混亂,異常也很容易被濫用。 所以在go語言中,為了防止異常被濫用。我們常常使用函數(shù)的返回值來返回錯誤,而不是用異常來代替錯誤。如果在一些場景下確實需要處理異常,就可以使用panic和recover。panic用來拋出異常,recover用來恢復(fù)異常。
panic是Go語言中,用于終止程序的一種函數(shù),往往用在下面兩種情況:1)程序出現(xiàn)了很大的故障,例如不能在提供服務(wù)了。2)程序在運行階段碰到了內(nèi)存異常的操作,例如空指針的取值,改寫只讀內(nèi)存等。對于panic來說,1)場景往往是主動調(diào)用;2)場景則是被動調(diào)用,panic一旦產(chǎn)生之后,會將堆棧里面的數(shù)據(jù)dump出來,這樣就方便了開發(fā)人員來定位問題。recover是用來截獲panic異常信息的,截獲了之后,可以控制程序跳過panic的地方繼續(xù)執(zhí)行。
需要注意:
- panic 能夠改變程序的控制流,調(diào)用 panic 后會立刻停止執(zhí)行當前函數(shù)的剩余代碼,并在當前 Goroutine 中遞歸執(zhí)行調(diào)用方的 defer;
- recover 可以中止 panic 造成的程序崩潰。它是一個只能在 defer 中發(fā)揮作用的函數(shù),在其他作用域中調(diào)用不會發(fā)揮作用;
1.特性
- panic 只會觸發(fā)當前goroutine的defer
- revoce 只有在defer中調(diào)用才能生效
- panic 允許在defer中嵌套多磁調(diào)用
2.panic觸發(fā)流程
- 1.如果函數(shù)F中書寫并觸發(fā)了panic語句,會終止其后要執(zhí)行的代碼。在panic所在函數(shù)F內(nèi)如果存在要執(zhí)行的defer函數(shù)列表,則按照defer書寫順序的逆序執(zhí)行;
- 2.如果函數(shù)G調(diào)用函數(shù)F,則函數(shù)F panic后返回調(diào)用者函數(shù)G。函數(shù)G中,調(diào)用函數(shù)F語句之后的語句都不會執(zhí)行。假如函數(shù)G中也有要執(zhí)行的defer函數(shù)列表,則按照defer書寫順序的逆序子還行;
- 退出整個goroutine,并報告錯誤。
3.recover使用要點
- recover的作用是捕獲panic,從而恢復(fù)正常代碼執(zhí)行;
- recover必須配合defer使用;
- recover沒有傳入?yún)?shù),但是有返回值,返回值就是panic傳遞的值
4.使用場景
一般情況下有兩種情況用到:
- 程序遇到無法執(zhí)行下去的錯誤時,拋出錯誤,主動結(jié)束運行。
- 在調(diào)試程序時,通過 panic 來打印堆棧,方便定位錯誤。
一、實踐
1.跨線程失效
package main import ( "fmt" "time" ) func main() { // 主線程中的defer函數(shù)并不會執(zhí)行,因為子協(xié)程 panic后,主線程中的defer并不會執(zhí)行 defer println("in main") go func() { defer println("in goroutine") fmt.Println("子協(xié)程running") panic("子協(xié)程崩潰") }() time.Sleep(1 * time.Second) }
# 輸出 $ go run main.go 子協(xié)程running in goroutine panic: 子協(xié)程崩潰 goroutine 6 [running]: main.main.func1()
當運行這段代碼時會發(fā)現(xiàn) main 函數(shù)中的 defer 語句并沒有執(zhí)行,執(zhí)行的只有當前 Goroutine 中的 defer。
2.不起作用的recover
初學(xué) Go 語言工程師可能會寫出下面的代碼,在主程序中調(diào)用 recover 試圖中止程序的崩潰,但是從運行的結(jié)果中也能看出,下面的程序沒有正常退出。
package main import "fmt" func main() { defer fmt.Println("in main") if err := recover(); err != nil { fmt.Println(err) } panic("unknown err") }
# 輸出 $ go run main.go in main panic: unknown err goroutine 1 [running]: main.main() D:/gopath/src/Go_base/lesson/panic/demo5.go:11 +0x125
仔細分析一下這個過程就能理解這種現(xiàn)象背后的原因,recover 只有在發(fā)生 panic 之后調(diào)用才會生效。然而在上面的控制流中,recover 是在 panic 之前調(diào)用的,并不滿足生效的條件,所以我們需要在 defer 中使用 recover 關(guān)鍵字。
正確的寫法應(yīng)該是這樣:
package main import "fmt" func main() { defer fmt.Println("in main") defer func() { if err := recover(); err != nil { fmt.Println("occur error") fmt.Println(err) } }() panic("unknown err") }
3.嵌套使用panic
panic 是可以多次嵌套調(diào)用的。,如下所示的代碼就展示了如何在 defer 函數(shù)中多次調(diào)用 panic:
package main import "fmt" func main() { defer fmt.Println("in main") defer func() { defer func() { panic("panic again and again") }() panic("panic again") }() panic("panic once") }
# 輸出 $ go run main.go in main panic: panic once panic: panic again panic: panic again and again goroutine 1 [running]: main.main.func1.1()
從上述程序輸出的結(jié)果,我們可以確定程序多次調(diào)用 panic 也不會影響 defer 函數(shù)的正常執(zhí)行,所以使用 defer 進行收尾工作一般來說都是安全的。
4.注意事項
1.recover 語法
//以下捕獲失敗 defer recover() defer fmt.Prinntln(recover) defer func(){ func(){ recover() //無效,嵌套兩層 }() }() //以下捕獲有效 defer func(){ recover() }() func except(){ recover() } func test(){ defer except() panic("runtime error") }
2.多個panic只會捕捉最后一個
package main import "fmt" func main(){ defer func(){ if err := recover() ; err != nil { fmt.Println(err) } }() defer func(){ panic("three") }() defer func(){ panic("two") }() panic("one") }
小結(jié)
到此這篇關(guān)于Go中的 panic / recover 簡介與實踐的文章就介紹到這了,更多相關(guān)go panic / recover 內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
- Go panic和recover函數(shù)使用細節(jié)深入探究
- 一文帶你掌握Golang中panic與recover的使用方法
- GoLang中panic與recover函數(shù)以及defer語句超詳細講解
- Golang中panic與recover的區(qū)別
- Golang異常處理之defer,panic,recover的使用詳解
- Golang 錯誤捕獲Panic與Recover的使用
- 小學(xué)生也能看懂的Golang異常處理recover panic
- Go中recover與panic區(qū)別詳解
- go語言的panic和recover函數(shù)用法實例
- go語言異常panic和恢復(fù)recover用法實例
- Go語言panic和recover的用法實例
相關(guān)文章
利用GoLang?Fiber進行高性能Web開發(fā)實例詳解
這篇文章主要為大家介紹了利用GoLang?Fiber進行高性能Web開發(fā)實例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪2024-01-01Go語言基礎(chǔ)Json序列化反序列化及文件讀寫示例詳解
這篇文章主要為大家介紹了Go語言基礎(chǔ)Json序列化反序列化以及文件讀寫的示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助2021-11-11