天天看點

golang捕獲異常

Go語言追求簡潔優雅,是以,Go語言不支援傳統的 try…catch…finally 這種異常,因為Go語言的設計者們認為,将異常與控制結構混在一起會很容易使得代碼變得混亂。因為開發者很容易濫用異常,甚至一個小小的錯誤都抛出一個異常。在Go語言中,使用多值傳回來傳回錯誤。不要用異常代替錯誤,更不要用來控制流程。在極個别的情況下,也就是說,遇到真正的異常的情況下(比如除數為0了)。才使用Go中引入的Exception處理:defer, panic, recover。

這幾個異常的使用場景可以這麼簡單描述:Go中可以抛出一個panic的異常,然後在defer中通過recover捕獲這個異常,然後正常處理。

例子代碼:

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是動态的。

<code>func</code> <code>f() (result int) {</code>

<code>  </code><code>defer</code><code>func</code><code>() {</code>

<code>    </code><code>result++</code>

<code>  </code><code>}()</code>

<code>  </code><code>return</code><code>0</code>

<code>}</code>

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

上面的函數就傳回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

package main  

//實作 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 函數裡面,不能放到子函數。

上一篇: go tool pprof

繼續閱讀