Go語言追求簡潔優(yōu)雅,所以,Go語言不支持傳統(tǒng)的 try…catch…finally 這種異常,因為Go語言的設計者們認為,將異常與控制結(jié)構(gòu)混在一起會很容易使得代碼變得混亂。因為開發(fā)者很容易濫用異常,甚至一個小小的錯誤都拋出一個異常。在Go語言中,使用多值返回來返回錯誤。不要用異常代替錯誤,更不要用來控制流程。在極個別的情況下,也就是說,遇到真正的異常的情況下(比如除數(shù)為0了)。才使用Go中引入的Exception處理:defer, panic, recover。
10年積累的網(wǎng)站制作、成都網(wǎng)站制作經(jīng)驗,可以快速應對客戶對網(wǎng)站的新想法和需求。提供各種問題對應的解決方案。讓選擇我們的客戶得到更好、更有力的網(wǎng)絡服務。我雖然不認識你,你也不認識我。但先建設網(wǎng)站后付款的網(wǎng)站建設流程,更有通渭免費網(wǎng)站建設讓你可以放心的選擇與我們合作。
這幾個異常的使用場景可以這么簡單描述: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其實就是panic傳入的內(nèi)容,55
}
fmt.Println("d")
}()
f()
}
func f(){
fmt.Println("a")
panic(55)
fmt.Println("b")
fmt.Println("f")
}
輸出結(jié)果:
a
c
55
d
exit code 0, process exited normally.
參考: http://blog.csdn.net/ghost911_slb/article/details/7831574
defer 英文原意: vi. 推遲;延期;服從 vt. 使推遲;使延期。
defer的思想類似于C++中的析構(gòu)函數(shù),不過Go語言中“析構(gòu)”的不是對象,而是函數(shù),defer就是用來添加函數(shù)結(jié)束時執(zhí)行的語句。注意這里強調(diào)的是添加,而不是指定,因為不同于C++中的析構(gòu)函數(shù)是靜態(tài)的,Go中的defer是動態(tài)的。
func
f() (result int) {
defer
func
()
{
result++
}()
return
0
}
上面函數(shù)返回1,因為defer中添加了一個函數(shù),在函數(shù)返回前改變了命名返回值的值。是不是很好用呢。但是,要注意的是,如果我們的defer語句沒有執(zhí)行,那么defer的函數(shù)就不會添加,如果把上面的程序改成這樣:
func
f() (result int) {
return
0
defer
func
()
{
result++
}()
return
0
}
上面的函數(shù)就返回0了,因為還沒來得及添加defer的東西,函數(shù)就返回了。
另外值得一提的是,defer可以多次,這樣形成一個defer棧,后defer的語句在函數(shù)返回時將先被調(diào)用。
參考: http://weager.sinaapp.com/?p=31
panic 英文原意:n. 恐慌,驚慌;大恐慌 adj. 恐慌的;沒有理由的 vt. 使恐慌 vi. 十分驚慌
panic 是用來表示非常嚴重的不可恢復的錯誤的。在Go語言中這是一個內(nèi)置函數(shù),接收一個interface{}類型的值(也就是任何值了)作為參數(shù)。panic的作用就像我們平常接觸的異常。不過Go可沒有try…catch,所以,panic一般會導致程序掛掉(除非recover)。所以,Go語言中的異常,那真的是異常了。你可以試試,調(diào)用panic看看,程序立馬掛掉,然后Go運行時會打印出調(diào)用棧。
但是,關鍵的一點是,即使函數(shù)執(zhí)行的時候panic了,函數(shù)不往下走了,運行時并不是立刻向上傳遞panic,而是到defer那,等defer的東西都跑完了,panic再向上傳遞。所以這時候 defer 有點類似 try-catch-finally 中的 finally。
panic就是這么簡單。拋出個真正意義上的異常。
recover 英文原意: vt. 恢復;彌補;重新獲得 vi. 恢復;勝訴;重新得球 n. 還原至預備姿勢
上面說到,panic的函數(shù)并不會立刻返回,而是先defer,再返回。這時候(defer的時候),如果有辦法將panic捕獲到,并阻止panic傳遞,那就異常的處理機制就完善了。
Go語言提供了recover內(nèi)置函數(shù),前面提到,一旦panic,邏輯就會走到defer那,那我們就在defer那等著,調(diào)用recover函數(shù)將會捕獲到當前的panic(如果有的話),被捕獲到的panic就不會向上傳遞了,于是,世界恢復了和平。你可以干你想干的事情了。
不過要注意的是,recover之后,邏輯并不會恢復到panic那個點去,函數(shù)還是會在defer之后返回。
用Go實現(xiàn)類似 try catch 的異常處理有個例子在:
[plain] view plain copy
package main
//實現(xiàn) try catch 例子
func Try(fun func(), handler func(interface{})) {
defer func() {
if err := recover(); err != nil {
handler(err)
}
}()
fun()
}
func main() {
Try(func() {
panic("foo")
}, func(e interface{}) {
print(e)
})
}
注意:
recover 只能放到 defer 函數(shù)里面,不能放到子函數(shù)。