Go的異常處理 defer, panic, recover

Go語言追求簡潔優雅,所以,Go語言不支持傳統的 try…catch…finally 這種異常,因爲Go語言的設計者們認爲,將異常與控制結構混在一起會很容易使得代碼變得混亂。因爲開發者很容易濫用異常,甚至一個小小的錯誤都拋出一個異常。在Go語言中,使用多值返回來返回錯誤。不要用異常代替錯誤,更不要用來控制流程。在極個別的情況下,也就是說,遇到真正的異常的情況下(比如除數爲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其實就是panic傳入的內容,55
        }
        fmt.Println("d")
    }()
    f()
}
 
func f(){
    fmt.Println("a")
    panic(55)
    fmt.Println("b")
    fmt.Println("f")
}
輸出結果:

a
c
55
d
exit code 0, process exited normally.

參考: http://blog.csdn.net/ghost911_slb/article/details/7831574

 

defer

defer 英文原意: vi. 推遲;延期;服從   vt. 使推遲;使延期。

defer的思想類似於C++中的析構函數,不過Go語言中“析構”的不是對象,而是函數,defer就是用來添加函數結束時執行的語句。注意這裏強調的是添加,而不是指定,因爲不同於C++中的析構函數是靜態的,Go中的defer是動態的。

func f() (result int) {

  deferfunc() {
    result++
  }()
  return0
}

上面函數返回1,因爲defer中添加了一個函數,在函數返回前改變了命名返回值的值。是不是很好用呢。但是,要注意的是,如果我們的defer語句沒有執行,那麼defer的函數就不會添加,如果把上面的程序改成這樣:

func f() (result int) {

  return0
  deferfunc() {
    result++
  }()
  return0
}

上面的函數就返回0了,因爲還沒來得及添加defer的東西,函數就返回了。

另外值得一提的是,defer可以多次,這樣形成一個defer棧,後defer的語句在函數返回時將先被調用。

參考: http://weager.sinaapp.com/?p=31 

 

panic

panic 英文原意:n. 恐慌,驚慌;大恐慌  adj. 恐慌的;沒有理由的  vt. 使恐慌  vi. 十分驚慌

panic 是用來表示非常嚴重的不可恢復的錯誤的。在Go語言中這是一個內置函數,接收一個interface{}類型的值(也就是任何值了)作爲參數。panic的作用就像我們平常接觸的異常。不過Go可沒有try…catch,所以,panic一般會導致程序掛掉(除非recover)。所以,Go語言中的異常,那真的是異常了。你可以試試,調用panic看看,程序立馬掛掉,然後Go運行時會打印出調用棧。
但是,關鍵的一點是,即使函數執行的時候panic了,函數不往下走了,運行時並不是立刻向上傳遞panic,而是到defer那,等defer的東西都跑完了,panic再向上傳遞。所以這時候 defer 有點類似 try-catch-finally 中的 finally。
panic就是這麼簡單。拋出個真正意義上的異常。

 

recover

recover 英文原意: vt. 恢復;彌補;重新獲得   vi. 恢復;勝訴;重新得球   n. 還原至預備姿勢

上面說到,panic的函數並不會立刻返回,而是先defer,再返回。這時候(defer的時候),如果有辦法將panic捕獲到,並阻止panic傳遞,那就異常的處理機制就完善了。

Go語言提供了recover內置函數,前面提到,一旦panic,邏輯就會走到defer那,那我們就在defer那等着,調用recover函數將會捕獲到當前的panic(如果有的話),被捕獲到的panic就不會向上傳遞了,於是,世界恢復了和平。你可以幹你想幹的事情了。

不過要注意的是,recover之後,邏輯並不會恢復到panic那個點去,函數還是會在defer之後返回。

 

用Go實現類似 try catch 的異常處理有個例子在:

[plain] view plain copy
  1. package main  
  2.   
  3. //實現 try catch 例子  
  4. func Try(fun func(), handler func(interface{})) {  
  5.     defer func() {  
  6.         if err := recover(); err != nil {  
  7.             handler(err)  
  8.         }  
  9.     }()  
  10.     fun()  
  11. }  
  12.   
  13. func main() {  
  14.     Try(func() {  
  15.        panic("foo")  
  16.     }, func(e interface{}) {  
  17.        print(e)  
  18.     })  
  19. }  

 

結論:

Go對待異常(準確的說是panic)的態度就是這樣,沒有全面否定異常的存在,同時極力不鼓勵多用異常。

參考:http://blog.dccmx.com/2012/01/exception-the-go-way/

http://kejibo.com/golang-exceptions-handle-defer-try/

http://bookjovi.iteye.com/blog/1335282

https://github.com/astaxie/build-web-application-with-golang/blob/master/02.3.md


轉載路徑:http://www.cnblogs.com/ghj1976/archive/2013/02/11/2910114.html

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章