Go中的 panic / recover 簡(jiǎn)介與實(shí)踐記錄
簡(jiǎn)介
go語(yǔ)言追求簡(jiǎn)潔,所以go語(yǔ)言中沒(méi)有try…catch語(yǔ)句。因?yàn)間o語(yǔ)言的作者認(rèn)為將異常和控制語(yǔ)句混在一起,很容易讓這個(gè)程序變得混亂,異常也很容易被濫用。 所以在go語(yǔ)言中,為了防止異常被濫用。我們常常使用函數(shù)的返回值來(lái)返回錯(cuò)誤,而不是用異常來(lái)代替錯(cuò)誤。如果在一些場(chǎng)景下確實(shí)需要處理異常,就可以使用panic和recover。panic用來(lái)拋出異常,recover用來(lái)恢復(fù)異常。
panic是Go語(yǔ)言中,用于終止程序的一種函數(shù),往往用在下面兩種情況:1)程序出現(xiàn)了很大的故障,例如不能在提供服務(wù)了。2)程序在運(yùn)行階段碰到了內(nèi)存異常的操作,例如空指針的取值,改寫只讀內(nèi)存等。對(duì)于panic來(lái)說(shuō),1)場(chǎng)景往往是主動(dòng)調(diào)用;2)場(chǎng)景則是被動(dòng)調(diào)用,panic一旦產(chǎn)生之后,會(huì)將堆棧里面的數(shù)據(jù)dump出來(lái),這樣就方便了開發(fā)人員來(lái)定位問(wèn)題。recover是用來(lái)截獲panic異常信息的,截獲了之后,可以控制程序跳過(guò)panic的地方繼續(xù)執(zhí)行。
需要注意:
- panic 能夠改變程序的控制流,調(diào)用 panic 后會(huì)立刻停止執(zhí)行當(dāng)前函數(shù)的剩余代碼,并在當(dāng)前 Goroutine 中遞歸執(zhí)行調(diào)用方的 defer;
- recover 可以中止 panic 造成的程序崩潰。它是一個(gè)只能在 defer 中發(fā)揮作用的函數(shù),在其他作用域中調(diào)用不會(huì)發(fā)揮作用;
1.特性
- panic 只會(huì)觸發(fā)當(dāng)前goroutine的defer
- revoce 只有在defer中調(diào)用才能生效
- panic 允許在defer中嵌套多磁調(diào)用
2.panic觸發(fā)流程
- 1.如果函數(shù)F中書寫并觸發(fā)了panic語(yǔ)句,會(huì)終止其后要執(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語(yǔ)句之后的語(yǔ)句都不會(huì)執(zhí)行。假如函數(shù)G中也有要執(zhí)行的defer函數(shù)列表,則按照defer書寫順序的逆序子還行;
- 退出整個(gè)goroutine,并報(bào)告錯(cuò)誤。
3.recover使用要點(diǎn)
- recover的作用是捕獲panic,從而恢復(fù)正常代碼執(zhí)行;
- recover必須配合defer使用;
- recover沒(méi)有傳入?yún)?shù),但是有返回值,返回值就是panic傳遞的值
4.使用場(chǎng)景
一般情況下有兩種情況用到:
- 程序遇到無(wú)法執(zhí)行下去的錯(cuò)誤時(shí),拋出錯(cuò)誤,主動(dòng)結(jié)束運(yùn)行。
- 在調(diào)試程序時(shí),通過(guò) panic 來(lái)打印堆棧,方便定位錯(cuò)誤。
一、實(shí)踐
1.跨線程失效
package main import ( "fmt" "time" ) func main() { // 主線程中的defer函數(shù)并不會(huì)執(zhí)行,因?yàn)樽訁f(xié)程 panic后,主線程中的defer并不會(huì)執(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()
當(dāng)運(yùn)行這段代碼時(shí)會(huì)發(fā)現(xiàn) main 函數(shù)中的 defer 語(yǔ)句并沒(méi)有執(zhí)行,執(zhí)行的只有當(dāng)前 Goroutine 中的 defer。
2.不起作用的recover
初學(xué) Go 語(yǔ)言工程師可能會(huì)寫出下面的代碼,在主程序中調(diào)用 recover 試圖中止程序的崩潰,但是從運(yùn)行的結(jié)果中也能看出,下面的程序沒(méi)有正常退出。
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
仔細(xì)分析一下這個(gè)過(guò)程就能理解這種現(xiàn)象背后的原因,recover 只有在發(fā)生 panic 之后調(diào)用才會(huì)生效。然而在上面的控制流中,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 也不會(huì)影響 defer 函數(shù)的正常執(zhí)行,所以使用 defer 進(jìn)行收尾工作一般來(lái)說(shuō)都是安全的。
4.注意事項(xiàng)
1.recover 語(yǔ)法
//以下捕獲失敗 defer recover() defer fmt.Prinntln(recover) defer func(){ func(){ recover() //無(wú)效,嵌套兩層 }() }() //以下捕獲有效 defer func(){ recover() }() func except(){ recover() } func test(){ defer except() panic("runtime error") }
2.多個(gè)panic只會(huì)捕捉最后一個(gè)
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 簡(jiǎn)介與實(shí)踐的文章就介紹到這了,更多相關(guān)go panic / recover 內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
- Go panic和recover函數(shù)使用細(xì)節(jié)深入探究
- 一文帶你掌握Golang中panic與recover的使用方法
- GoLang中panic與recover函數(shù)以及defer語(yǔ)句超詳細(xì)講解
- Golang中panic與recover的區(qū)別
- Golang異常處理之defer,panic,recover的使用詳解
- Golang 錯(cuò)誤捕獲Panic與Recover的使用
- 小學(xué)生也能看懂的Golang異常處理recover panic
- Go中recover與panic區(qū)別詳解
- go語(yǔ)言的panic和recover函數(shù)用法實(shí)例
- go語(yǔ)言異常panic和恢復(fù)recover用法實(shí)例
- Go語(yǔ)言panic和recover的用法實(shí)例
相關(guān)文章
Go語(yǔ)言常見錯(cuò)誤之將接口定義在實(shí)現(xiàn)方
在Go中,接口起到一個(gè)十分關(guān)鍵的角色,它們提供了一種方式來(lái)定義對(duì)象的行為,而不需要知道對(duì)象的具體實(shí)現(xiàn),一個(gè)常見的錯(cuò)誤是在實(shí)現(xiàn)方而不是使用方定義接口,本文將詳細(xì)探討為何這樣做是一個(gè)錯(cuò)誤,以及如何避免它2024-01-01利用GoLang?Fiber進(jìn)行高性能Web開發(fā)實(shí)例詳解
這篇文章主要為大家介紹了利用GoLang?Fiber進(jìn)行高性能Web開發(fā)實(shí)例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2024-01-01Go語(yǔ)言基礎(chǔ)go install命令使用示例詳解
這篇文章主要為大家介紹了Go語(yǔ)言基礎(chǔ)go install命令的使用示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2021-11-11Go語(yǔ)言基礎(chǔ)Json序列化反序列化及文件讀寫示例詳解
這篇文章主要為大家介紹了Go語(yǔ)言基礎(chǔ)Json序列化反序列化以及文件讀寫的示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助2021-11-11詳解Go語(yǔ)言各種常見類型的默認(rèn)值和判空方法
本文主要介紹了詳解Go語(yǔ)言各種常見類型的默認(rèn)值和判空方法,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2023-04-04