Panic和Recover
Go沒有像Java那樣的異常機制,它不能拋出異常,而是使用了panic
和recover
機制。一定要記住,你應當把它作為最后的手段來使用,也就是說,你的代碼中應當沒有,或者很少有panic
的東西。這是個強大的工具,請明智地使用它。那么,我們應該如何使用它呢?
Panic
是一個內建函數,可以中斷原有的控制流程,進入一個令人恐慌的流程中。當函數F
調用panic
,函數F
的執行被中斷,但是F
中的延遲函數會正常執行,然后F
返回到調用它的地方。在調用的地方,F
的行為就像調用了panic
。這一過程繼續向上,直到發生panic
的goroutine
中所有調用的函數返回,此時程序退出。恐慌可以直接調用panic
產生。也可以由運行時錯誤產生,例如訪問越界的數組。
Recover
是一個內建的函數,可以讓進入令人恐慌的流程中的goroutine
恢復過來。recover
僅在延遲函數中有效。在正常的執行過程中,調用recover
會返回nil
,并且沒有其它任何效果。如果當前的goroutine
陷入恐慌,調用recover
可以捕獲到panic
的輸入值,并且恢復正常的執行。
下面這個函數演示了如何在過程中使用panic
var user = os.Getenv("USER") func init() { if user == "" { panic("no value for $USER") } }
下面這個函數檢查作為其參數的函數在執行時是否會產生panic
:
func throwsPanic(f func()) (b bool) { defer func() { if x := recover(); x != nil { b = true } }() f() //執行函數f,如果f中出現了panic,那么就可以恢復回來 return }
最容易理解就是給個例子,文章里有例子:
package main import( "fmt" //"os" ) var user = "" func inita() { defer func(){ fmt.Print("defer##\n") }() if user == "" { fmt.Print("@@@before panic\n") panic("no value for user\n") fmt.Print("!!after panic\n") } } func throwsPanic (f func()) (b bool){ defer func(){ if x:= recover(); x != nil{ fmt.Print(x) b = true } }() f() fmt.Print("after the func run") return } func main(){ throwsPanic(inita) }
執行結果:
D:\go>go run b.go
@@@before panic
defer##
no value for user
如上面所說的:
panic
在user=""
時,打斷了函數的執行,fmt.Print("!!after panic\n")
沒有執行。 但函數中的延遲函數會正常執行,打印了 defer##
。然后返回到調用該函數的地方,繼續上面的過程。
直到執行完所有函數的defer
,退出程序。Recover
可以捕獲到panic
的值,上面的打印no value for user
。并且恢復正常的執行。