Golang中錯誤處理機制詳解
Golang錯誤處理機制
panic異常
panic異常
- Go的類型系統(tǒng)會在編譯時捕獲很多錯誤,但有些錯誤只能在運行時檢查,比如除零錯誤、數(shù)組訪問越界、空指針引用等,這些運行時錯誤會引起panic異常。
- 當panic異常發(fā)生時,程序會中斷運行,并立即執(zhí)行當前函數(shù)中的defer語句,然后程序會從調(diào)用棧中逐級返回,返回同時執(zhí)行各級函數(shù)中的defer語句,直到panic被捕獲或程序終止。
例如,下面的代碼在運行過程中,由于存在除零錯誤將導致panic異常。如下:
package main import "fmt" func Division(num1 int, num2 int) int { defer fmt.Println("Division defer...") return num1 / num2 } func main() { defer fmt.Println("main defer...") // panic異常 result := Division(10, 0) fmt.Printf("result = %d\n", result) fmt.Println("other code...") }
運行代碼后可以看到,main函數(shù)中調(diào)用Division函數(shù)之后的代碼沒有被執(zhí)行。但在程序退出之前,Division和main函數(shù)中的defer語句被逐一執(zhí)行。如下:
recover捕獲異常
recover捕獲異常
- 當觸發(fā)panic異常時,我們希望程序可以繼續(xù)運行,并希望在觸發(fā)異常后,能夠執(zhí)行某些操作,比如給管理員發(fā)送一條告警信息。
- 在Go中,捕獲異常通過在defer中調(diào)用內(nèi)建函數(shù)recover實現(xiàn),recover函數(shù)會使程序從panic中恢復,并返回觸發(fā)panic的值,如果程序沒有發(fā)生panic,則recover調(diào)用將會返回nil。
- 需要注意的是,當異常被捕獲后,觸發(fā)panic異常的函數(shù)也不會繼續(xù)運行,但能正常返回。
例如,下面在Division函數(shù)的defer中捕獲了panic異常,并將觸發(fā)panic異常的值進行了返回。如下:
package main import "fmt" func Division(num1 int, num2 int) (result int, err error) { defer func() { if r := recover(); r != nil { err = r.(error) // 類型斷言,將r轉(zhuǎn)換為error類型(后續(xù)介紹) } }() return num1 / num2, nil } func main() { // recover捕獲異常 result, err := Division(10, 0) if err != nil { fmt.Printf("division operation error, err = %v\n", err) } else { fmt.Printf("result = %d\n", result) } }
運行代碼后可以看到,由于panic異常被捕獲使得Division函數(shù)正常返回,同時main函數(shù)后續(xù)的代碼也被正常執(zhí)行。如下:
由于panic異常會逐級拋出,直到異常被捕獲或程序終止,因此可以選擇在調(diào)用鏈的任意一個函數(shù)中對panic進行捕獲。如下:
package main import "fmt" func Division(num1 int, num2 int) int { return num1 / num2 } func main() { // recover捕獲異常 defer func() { if err := recover(); err != nil { fmt.Printf("recoverd from panic, err = %v\n", err) } }() result := Division(10, 0) fmt.Printf("result = %d\n", result) fmt.Println("other code...") }
由于Division函數(shù)中沒有對panic進行捕獲,導致main函數(shù)觸發(fā)panic異常,雖然在main函數(shù)中對panic進行了捕獲,但main函數(shù)的后續(xù)代碼將不會被執(zhí)行。如下:
自定義錯誤
自定義錯誤
- 不是只有程序在運行時出現(xiàn)錯誤才能觸發(fā)panic異常,通過調(diào)用內(nèi)建函數(shù)panic也能觸發(fā)panic異常。
- panic函數(shù)能夠接收任何類型的值作為參數(shù),并在觸發(fā)panic時將其作為觸發(fā)panic的值。
- 觸發(fā)panic的值通常是error類型的,通過errors包中的New函數(shù)可以創(chuàng)建一個error類型的錯誤值,該函數(shù)接收一個string類型的參數(shù),并返回創(chuàng)建的error類型的錯誤值。
例如,我們規(guī)定除數(shù)不能為20,否則觸發(fā)panic異常。如下:
package main import ( "errors" "fmt" ) func Division(num1 int, num2 int) (result int, err error) { defer func() { if r := recover(); r != nil { err = r.(error) } }() if num2 == 20 { panic(errors.New("integer divide by 20")) } return num1 / num2, nil } func main() { // recover捕獲異常 result, err := Division(10, 20) if err != nil { fmt.Printf("division operation error, err = %v\n", err) } else { fmt.Printf("result = %d\n", result) } }
運行代碼后可以看到,通過panic函數(shù)觸發(fā)的異常也能夠成功被捕獲。如下:
說明一下: 通過recover將panic捕獲后,可以繼續(xù)通過調(diào)用panic函數(shù)將panic再次拋出,使其傳播到更高層的調(diào)用棧進行進一步處理,這叫做異常的重新拋出。
到此這篇關(guān)于Golang中錯誤處理機制詳解的文章就介紹到這了,更多相關(guān)Golang錯誤處理機制內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Go語言實戰(zhàn)之實現(xiàn)一個簡單分布式系統(tǒng)
如今很多云原生系統(tǒng)、分布式系統(tǒng),例如?Kubernetes,都是用?Go?語言寫的,這是因為?Go?語言天然支持異步編程。本篇文章將介紹如何用?Go?語言編寫一個簡單的分布式系統(tǒng),需要的小伙伴開業(yè)跟隨小編一起學習一下2022-10-10Golang在Mac、Linux、Windows下如何交叉編譯的實現(xiàn)
這篇文章主要介紹了Golang在Mac、Linux、Windows下如何交叉編譯的實現(xiàn),文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧2020-03-03數(shù)據(jù)競爭和內(nèi)存重分配Golang slice并發(fā)不安全問題解決
這篇文章主要為大家介紹了數(shù)據(jù)競爭和內(nèi)存重分配Golang slice并發(fā)不安全問題解決,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪2023-10-10go-micro使用Consul做服務發(fā)現(xiàn)的方法和原理解析
這篇文章主要介紹了go-micro使用Consul做服務發(fā)現(xiàn)的方法和原理,這里提供一個通過docker快速安裝Consul的方式,當然前提是你得安裝了docker,需要的朋友可以參考下2022-04-04