Go語(yǔ)言異常處理(Panic和recovering)用法詳解
基本語(yǔ)法
異常處理是程序健壯性的關(guān)鍵,往往開(kāi)發(fā)人員的開(kāi)發(fā)經(jīng)驗(yàn)的多少?gòu)漠惓2糠痔幚砩暇湍艿玫襟w現(xiàn)。如何適度的添加異常,往往是整個(gè)產(chǎn)品體驗(yàn)成敗的關(guān)鍵。
Go語(yǔ)言中沒(méi)有Try Catch Exception機(jī)制,但是提供了panic-and-recover機(jī)制。
Panic
內(nèi)置函數(shù)panic()
類似raise,能夠停止正常的流程
當(dāng)函數(shù)內(nèi)調(diào)用panic,正常的流程將被終止,defer函數(shù)仍然會(huì)被執(zhí)行
Panic引起的原因可以是主動(dòng)調(diào)用,也可以是運(yùn)行時(shí)錯(cuò)誤,例如數(shù)組越界
Recover
內(nèi)置函數(shù),用于處理panic(panicking goroutine)
在defer函數(shù)中使用有意義,正常執(zhí)行過(guò)程中,recover只會(huì)返回nil,而且沒(méi)有其他副作用
如果當(dāng)前Goroutine(后面講到并發(fā)時(shí)會(huì)重點(diǎn)講解,這里就是理解成程序的執(zhí)行過(guò)程中)發(fā)生panicking,將自動(dòng)捕獲panic值給recover,同時(shí)恢復(fù)正常執(zhí)行
示例一:recover()使用方法
本示例主要說(shuō)明如何捕獲異常,先來(lái)模擬一種異常情況,在一些項(xiàng)目中,經(jīng)常有數(shù)組下標(biāo)越界的情況,我們來(lái)人為構(gòu)造一下
package main
import "fmt"
func MyPanic() {
fmt.Println("Running in MyPanic...")
var a[]int
a[3] = 5
}
func main() {
MyPanic()
}
此時(shí)運(yùn)行程序,很明顯會(huì)出現(xiàn)如下問(wèn)題:
Running in MyPanic...
panic: runtime error: index out of range [3] with length 0
goroutine 1 [running]:
main.MyPanic()
/root/workspace/go/test_panic_recover_basic.go:14 +0x5e
main.main()
/root/workspace/go/test_panic_recover_basic.go:18 +0x17
exit status 2
嘗試用recover進(jìn)行異常處理
package main
import "fmt"
func MyPanic() {
defer func() {
if x := recover(); x != nil {
fmt.Printf("[ERROR]: My panic handle error: %s\n", x)
}
}()
fmt.Println("Running in MyPanic...")
var a[]int
a[3] = 5
}
func main() {
MyPanic()
}
就上面的代碼進(jìn)行一下分析:
首先增加了一個(gè)defer函數(shù)
在defer中,使用x := recover(); x != nil方式捕獲異常,如果獲取的值不為空,則證明有異常發(fā)生
獲取異常信息時(shí),直接用剛剛的變量就可以輸出詳細(xì)的錯(cuò)誤信息,執(zhí)行結(jié)果如下所示
Running in MyPanic...
[ERROR]: My panic handle error: runtime error: index out of range [3] with length 0
實(shí)例二:panic()使用方法
本示例除了介紹panic(),還實(shí)現(xiàn)了一種統(tǒng)一的ErrorHandler方法(有點(diǎn)像Python中的裝飾器),來(lái)統(tǒng)一處理函數(shù)的異常
package main
import "fmt"
func ErrorHandler(f func()) (b bool) {
defer func() {
if x := recover(); x != nil {
fmt.Printf("[ERROR]Handle error here: %s\n", x)
b = true
}
}()
f()
return
}
func CallPanic() {
panic("Call panic")
}
func main() {
fmt.Println(ErrorHandler(CallPanic))
}
我們來(lái)對(duì)代碼進(jìn)行一下分析:
定義了兩個(gè)函數(shù),一個(gè)是CallPanic()產(chǎn)生異常,一個(gè)是ErrorHandler()來(lái)捕獲所有函數(shù)的異常
CallPanic()邏輯很簡(jiǎn)單,就是用panic()內(nèi)置函數(shù)產(chǎn)生異常,后面的參數(shù)就是異常的具體內(nèi)容
ErrorHandler的參數(shù)是一個(gè)函數(shù),也就是利用函數(shù)作為值的特性,而返回值為bool類型
ErrorHandler中對(duì)于異常捕獲與示例一種相同,利用defer函數(shù)完成,而在函數(shù)體內(nèi),執(zhí)行被調(diào)用的函數(shù)f()
從執(zhí)行結(jié)果來(lái)看,ErrorHandler中輸出了Call panic的異常和返回結(jié)果true
到此這篇關(guān)于Go語(yǔ)言異常處理(Panic和recovering)用法詳解的文章就介紹到這了,更多相關(guān)Go語(yǔ)言異常處理內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
go?sync?Waitgroup數(shù)據(jù)結(jié)構(gòu)實(shí)現(xiàn)基本操作詳解
這篇文章主要為大家介紹了go?sync?Waitgroup數(shù)據(jù)結(jié)構(gòu)實(shí)現(xiàn)基本操作詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-01-01
使用Go語(yǔ)言玩轉(zhuǎn) RESTful API 服務(wù)
RESTful API是一種基于HTTP協(xié)議的API設(shè)計(jì)風(fēng)格,遵循REST架構(gòu)風(fēng)格,這篇文章主要為大家介紹了如何通過(guò)Go語(yǔ)言構(gòu)建RESTful API服務(wù),有需要的可以了解下2025-02-02
Golang如何調(diào)用windows下的dll動(dòng)態(tài)庫(kù)中的函數(shù)
這篇文章主要介紹了Golang如何調(diào)用windows下的dll動(dòng)態(tài)庫(kù)中的函數(shù)方式,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2021-05-05
Golang HTTP 服務(wù)平滑重啟及升級(jí)的思路
Golang HTTP服務(wù)在上線時(shí),需要重新編譯可執(zhí)行文件,關(guān)閉正在運(yùn)行的進(jìn)程,然后再啟動(dòng)新的運(yùn)行進(jìn)程。這篇文章主要介紹了Golang HTTP 服務(wù)平滑重啟及升級(jí),需要的朋友可以參考下2020-04-04
go語(yǔ)言區(qū)塊鏈學(xué)習(xí)調(diào)用智能合約
這篇文章主要為大家介紹了go語(yǔ)言區(qū)塊鏈學(xué)習(xí)中如何調(diào)用智能合約的實(shí)現(xiàn)示例,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步2021-10-10
Golang常用環(huán)境變量說(shuō)明與設(shè)置詳解
這篇文章主要介紹了Golang常用環(huán)境變量說(shuō)明與設(shè)置,需要的朋友可以參考下2020-02-02
從Context到go設(shè)計(jì)理念輕松上手教程
這篇文章主要為大家介紹了從Context到go設(shè)計(jì)理念輕松上手教程詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2022-09-09

