Go語言追求簡潔優(yōu)雅,所以,Go語言不支持傳統(tǒng)的try…catch…finally 這種異常,因?yàn)镚o語言的設(shè)計(jì)者們認(rèn)為,將異常與控制結(jié)構(gòu)混在一起會很容易使得代碼變得混亂。因?yàn)殚_發(fā)者很容易濫用異常,甚至一個小小的錯誤都拋出一個異常。
在Go語言中,使用多值返回來返回錯誤。不要用異常代替錯誤,更不要用來控制流程。在極個別的情況下,也就是說,遇到真正的異常的情況下(比如除數(shù)為 0了)。才使用Go中引入的Exception處理:defer, panic, recover。
這幾個異常的使用場景可以這么簡單描述:Go中可以拋出一個panic的異常,然后在defer中通過recover捕獲這個異常,然后正常處理。
例子代碼:
package main import "fmt" func main(){ defer func(){ // 必須要先聲明defer,否則不能捕獲到panic異常 fmt.Println("c") if err:=recover();err!=nil{ fmt.Println(err) // 這里的err其實(shí)就是panic傳入的內(nèi)容,55 } fmt.Println("d") }() f() } func f(){ fmt.Println("a") panic(55) fmt.Println("b") fmt.Println("f") } 輸出結(jié)果: a c d exit code 0, process exited normally.
defer
defer 英文原意: vi. 推遲;延期;服從 vt. 使推遲;使延期。
defer的思想類似于C++中的析構(gòu)函數(shù),不過Go語言中“析構(gòu)”的不是對象,而是函數(shù),defer就是用來添加函數(shù)結(jié)束時執(zhí)行的語句。注意這里強(qiáng)調(diào)的是添加,而不是指定,因?yàn)椴煌贑++中的析構(gòu)函數(shù)是靜態(tài)的,Go中的defer是動態(tài)的。
func f() (result int) { defer func() { result++ }() return 0 }
上面的函數(shù)就返回0了,因?yàn)檫€沒來得及添加defer的東西,函數(shù)就返回了。
另外值得一提的是,defer可以多次,這樣形成一個defer棧,后defer的語句在函數(shù)返回時將先被調(diào)用。
panic
panic 是用來表示非常嚴(yán)重的不可恢復(fù)的錯誤的。在Go語言中這是一個內(nèi)置函數(shù),接收一個interface{}類型的值(也就是任何值了)作為參數(shù)。panic 的作用就像我們平常接觸的異常。
不過Go可沒有try…catch,所以,panic一般會導(dǎo)致程序掛掉(除非recover)。所以,Go語言中的異常,那真的是異常了。你可以試試,調(diào)用panic看看,程序立馬掛掉,然后Go運(yùn)行時會打印出調(diào)用棧。
但是,關(guān)鍵的一點(diǎn)是,即使函數(shù)執(zhí)行的時候 panic了,函數(shù)不往下走了,運(yùn)行時并不是立刻向上傳遞panic,而是到defer那,等defer的東西都跑完了,panic再向上傳遞。所以這時候 defer 有點(diǎn)類似 try-catch-finally 中的 finally。
recover
上面說到,panic的函數(shù)并不會立刻返回,而是先defer,再返回。這時候(defer的時候),如果有辦法將panic捕獲到,并阻止panic傳遞,那就異常的處理機(jī)制就完善了。
Go語言提供了recover內(nèi)置函數(shù),前面提到,一旦panic,邏輯就會走到defer那,那我們就在defer那等著,調(diào)用recover函數(shù)將會捕獲到當(dāng)前的panic(如果有的話),被捕獲到的panic就不會向上傳遞了,于是,世界恢復(fù)了和平。你可以干你想干的事情了。
不過要注意的是,recover之后,邏輯并不會恢復(fù)到panic那個點(diǎn)去,函數(shù)還是會在defer之后返回。