有的同窗看到Go和TryCatch一塊兒出現,內心可能會說,難道Go語言升級了,加入了try...catch語句。哈哈,其實Go語言從建立之初就沒打算加入try...catch語句,由於建立Go的那幫大爺認爲try...catch挺煩人的,若是濫用,會形成程序混亂,因此就不打算加入try...catch(之後加不加入很差說)。java
既然Go語言中並無try...catch語句,那麼爲什麼文章標題說要使用TryCatch呢?其實Go語言中只是沒有try...catch語句,並非沒有異常處理機制。Go語言中的異常處理機制就是著名的異常三劍客:panic、defer和recover。經過這3個傢伙,是徹底能夠模擬出try...catch語句效果的,對了,後面還應該有個finally。在正式模擬try...catch語句以前,先來回顧下Go語言中的異常處理機制是如何玩的。函數
Go語言中的異常處理機制
在前面提到,Go語言經過panic、defer和recover來處理異常的,那麼這3個東西是什麼呢?this
不論是什麼異常處理機制,核心的原理都是同樣的,一般來說,一個完善的異常處理機制須要由下面3部分組成。spa
- 拋出異常
- 處理異常的代碼段
- 獲取異常信息
下面先用Java的異常處理機制來講明這一點。3d
import java.io.IOException; public class Main { public static void main(String[] args) { try { boolean ioException = false; if (ioException) { throw new IOException("ioexception"); } else { throw new Exception("exception"); } } catch (IOException e) { System.err.println(e); } catch (Exception e) { System.out.println(e); } finally { System.out.println("finally"); } } }
上面的代碼是標準的Java異常處理機制,try部分的throw用於拋出異常,而catch部分的代碼段用於處理特定的異常,經過catch子句的參數e能夠獲取異常信息。因此對於Java來講,上述的3個異常重要的組成部分都有。code
對於Go語言來講,panic、defer和recover也分別對應了這3部分。其中panic是一個函數,用於拋出異常,至關於Java中的throw函數。defer是一個關鍵字,用於修飾函數,用defer修飾的函數,在拋出異常時會自動調用。recover是一個函數,用於獲取異常信息,一般在用defer修飾的函數中使用。blog
下面是一段用Go語言處理異常的代碼。string
package main import "fmt" func main(){ // 處理異常的函數 defer func(){ fmt.Println("開始處理異常") // 獲取異常信息 if err:=recover();err!=nil{ // 輸出異常信息 fmt.Println("error:",err) } fmt.Println("結束異常處理") }() exceptionFun() } func exceptionFun(){ fmt.Println("exceptionFun開始執行") panic("異常信息") fmt.Println("exceptionFun執行結束") }
實現Go版的TryCatch
如今已經瞭解了Go語言的異常處理機制,那麼接下來使用異常處理機制來模擬try...catch...finally語句。io
如今來分析一下若是模擬。模擬的過程須要完成下面的工做。class
- try、catch和finally這3部分都有各自的代碼段,因此爲了模擬try...catch...finally,須要用3個Go函數來分別模擬try、catch和finally部分的代碼段。這3個Go函數是Try、Catch和Finally。
- 要肯定這3個函數在什麼地方調用。Try是正常執行的代碼,因此在要首先調用Try函數。而Catch函數只有在拋出異常時調用,因此應該在用defer修飾的函數中調用,並且須要在Catch函數中獲取異常信息,因此應該在使用cover函數獲取異常信息後再調用Catch函數,一般會將異常信息直接做爲參數傳遞給Catch函數。不論是否拋出異常,Finally函數都必須調用,因此應該用defer修飾Finally函數,並且是第1個用defer修飾的函數。這樣,在當前函數結束以前必定剛回調用Finally函數。
- 觸發異常,這就很是簡單了,直接用panic函數便可。
上面清楚地描述了用Go語言的異常處理機制模擬try...catch...finally語句的基本原理,下面給出完整的實現代碼。
package main import ( "fmt" ) type ExceptionStruct struct { Try func() Catch func(Exception) Finally func() } type Exception interface{} func Throw(up Exception) { panic(up) } func (this ExceptionStruct) Do() { if this.Finally != nil { defer this.Finally() } if this.Catch != nil { defer func() { if e := recover(); e != nil { this.Catch(e) } }() } this.Try() } func main() { fmt.Println("開始執行...") ExceptionStruct{ Try: func() { fmt.Println("try...") Throw("發生了錯誤") }, Catch: func(e Exception) { fmt.Printf("exception %v\n", e) }, Finally: func() { fmt.Println("Finally...") }, }.Do() fmt.Println("結束運行") }
上面的代碼將Try、Catch、Finally函數都封裝在了ExceptionStruct結構體中。而後調用方式就與前面的描述的一致了。執行這段代碼,會輸出以下圖的信息。
加強版的TryCatch
到如今爲止,其實已經完整地實現了try...catch...finally語句,但細心的同窗會發現,這個實現有一點小問題。一般的try...catch...finally語句,try部分有且只有1個,finally部分是可選的,但最多隻能有1個,而catch部分也是可選的,能夠有0到n個,也就是catch部分能夠有任意多個。但前面的實現,Catch函數只能指定一個,若是要指定任意多個應該如何作呢?其實很簡單,用一個Catch函數集合保存全部指定的Catch函數便可。不過須要快速定位某一個Catch函數。在Java中,是經過異常類型(如IOException、Exception等)定位特定的catch子句的,咱們也能夠模擬這一過程,經過特定的異常來定位與該異常對應的Catch函數,爲了方便,能夠用int類型的異常代碼。那麼在調用Catch函數以前,就須要經過異常代碼先定位到某一個Catch函數,而後再調用。下面就是完整的實現代碼。
package main import ( "log" ) type Exception struct { Id int // exception id Msg string // exception msg } type TryStruct struct { catches map[int]ExceptionHandler try func() } func Try(tryHandler func()) *TryStruct { tryStruct := TryStruct{ catches: make(map[int]ExceptionHandler), try: tryHandler, } return &tryStruct } type ExceptionHandler func(Exception) func (this *TryStruct) Catch(exceptionId int, catch func(Exception)) *TryStruct { this.catches[exceptionId] = catch return this } func (this *TryStruct) Finally(finally func()) { defer func() { if e := recover(); nil != e { exception := e.(Exception) if catch, ok := this.catches[exception.Id]; ok { catch(exception) } finally() } }() this.try() } func Throw(id int, msg string) Exception { panic(Exception{id,msg}) } func main() { exception.Try(func() { log.Println("try...") // 指定了異常代碼爲2,錯誤信息爲error2 exception.Throw(2,"error2") }).Catch(1, func(e exception.Exception) { log.Println(e.Id,e.Msg) }).Catch(2, func(e exception.Exception) { log.Println(e.Id,e.Msg) }).Finally(func() { log.Println("finally") }) }
執行結果以下圖所示。
這個實現與Java中的try...catch...finally的惟一區別就是必需要調用Finally函數,由於處理異常的代碼都在Finally函數中。不過這並不影響使用,若是finally部分沒什麼須要處理的,那麼就設置一個空函數便可。
爲了方便你們,我已經將該實現封裝成了函數庫,調用代碼以下:
package main import ( "exception" "log" ) func main() { exception.Try(func() { log.Println("try...") exception.Throw(2,"error2") }).Catch(1, func(e exception.Exception) { log.Println(e.Id,e.Msg) }).Catch(2, func(e exception.Exception) { log.Println(e.Id,e.Msg) }).Finally(func() { log.Println("finally") }) }
得到本文源代碼,請關注」極客起源「或」歐瑞科技「公衆號,並輸入308178得到源代碼。