這篇文章主要講解了“如何用Go語言異常機制模擬TryCatch異常捕捉”,文中的講解內(nèi)容簡單清晰,易于學(xué)習(xí)與理解,下面請大家跟著小編的思路慢慢深入,一起來研究和學(xué)習(xí)“如何用Go語言異常機制模擬TryCatch異常捕捉”吧!
創(chuàng)新互聯(lián)建站企業(yè)建站,10多年網(wǎng)站建設(shè)經(jīng)驗,專注于網(wǎng)站建設(shè)技術(shù),精于網(wǎng)頁設(shè)計,有多年建站和網(wǎng)站代運營經(jīng)驗,設(shè)計師為客戶打造網(wǎng)絡(luò)企業(yè)風(fēng)格,提供周到的建站售前咨詢和貼心的售后服務(wù)。對于成都網(wǎng)站建設(shè)、成都網(wǎng)站制作中不同領(lǐng)域進行深入了解和探索,創(chuàng)新互聯(lián)在網(wǎng)站建設(shè)中充分了解客戶行業(yè)的需求,以靈動的思維在網(wǎng)頁中充分展現(xiàn),通過對客戶行業(yè)精準市場調(diào)研,為客戶提供的解決方案。
不管是什么異常處理機制,核心的原理都是一樣的,通常來講,一個完善的異常處理機制需要由下面3部分組成。
拋出異常
處理異常的代碼段
獲取異常信息
下面先用Java的異常處理機制來說明這一點。
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子句的參數(shù)e可以獲取異常信息。所以對于Java來說,上述的3個異常重要的組成部分都有。
對于Go語言來說,panic、defer和recover也分別對應(yīng)了這3部分。其中panic是一個函數(shù),用于拋出異常,相當(dāng)于Java中的throw函數(shù)。defer是一個關(guān)鍵字,用于修飾函數(shù),用defer修飾的函數(shù),在拋出異常時會自動調(diào)用。recover是一個函數(shù),用于獲取異常信息,通常在用defer修飾的函數(shù)中使用。
下面是一段用Go語言處理異常的代碼。
package main
import "fmt"
func main(){
// 處理異常的函數(shù)
defer func(){
fmt.Println("開始處理異常")
// 獲取異常信息
if err:=recover();err!=nil{
// 輸出異常信息
fmt.Println("error:",err)
}
fmt.Println("結(jié)束異常處理")
}()
exceptionFun()
}
func exceptionFun(){
fmt.Println("exceptionFun開始執(zhí)行")
panic("異常信息")
fmt.Println("exceptionFun執(zhí)行結(jié)束")
}
實現(xiàn)Go版的TryCatch
現(xiàn)在已經(jīng)了解了Go語言的異常處理機制,那么接下來使用異常處理機制來模擬try...catch...finally語句。
現(xiàn)在來分析一下如果模擬。模擬的過程需要完成下面的工作。
try、catch和finally這3部分都有各自的代碼段,所以為了模擬try...catch...finally,需要用3個Go函數(shù)來分別模擬try、catch和finally部分的代碼段。這3個Go函數(shù)是Try、Catch和Finally。
要確定這3個函數(shù)在什么地方調(diào)用。Try是正常執(zhí)行的代碼,所以在要首先調(diào)用Try函數(shù)。而Catch函數(shù)只有在拋出異常時調(diào)用,所以應(yīng)該在用defer修飾的函數(shù)中調(diào)用,而且需要在Catch函數(shù)中獲取異常信息,所以應(yīng)該在使用cover函數(shù)獲取異常信息后再調(diào)用Catch函數(shù),通常會將異常信息直接作為參數(shù)傳遞給Catch函數(shù)。不管是否拋出異常,F(xiàn)inally函數(shù)都必須調(diào)用,所以應(yīng)該用defer修飾Finally函數(shù),而且是第1個用defer修飾的函數(shù)。這樣,在當(dāng)前函數(shù)結(jié)束之前一定剛回調(diào)用Finally函數(shù)。
觸發(fā)異常,這就非常簡單了,直接用panic函數(shù)即可。
上面清楚地描述了用Go語言的異常處理機制模擬try...catch...finally語句的基本原理,下面給出完整的實現(xiàn)代碼。
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("開始執(zhí)行...")
ExceptionStruct{
Try: func() {
fmt.Println("try...")
Throw("發(fā)生了錯誤")
},
Catch: func(e Exception) {
fmt.Printf("exception %v\n", e)
},
Finally: func() {
fmt.Println("Finally...")
},
}.Do()
fmt.Println("結(jié)束運行")
}
上面的代碼將Try、Catch、Finally函數(shù)都封裝在了ExceptionStruct結(jié)構(gòu)體中。然后調(diào)用方式就與前面的描述的一致了。執(zhí)行這段代碼,會輸出如下圖的信息。
image.png
增強版的TryCatch
到現(xiàn)在為止,其實已經(jīng)完整地實現(xiàn)了try...catch...finally語句,但細心的同學(xué)會發(fā)現(xiàn),這個實現(xiàn)有一點小問題。通常的try...catch...finally語句,try部分有且只有1個,finally部分是可選的,但最多只能有1個,而catch部分也是可選的,可以有0到n個,也就是catch部分可以有任意多個。但前面的實現(xiàn),Catch函數(shù)只能指定一個,如果要指定任意多個應(yīng)該如何做呢?其實很簡單,用一個Catch函數(shù)集合保存所有指定的Catch函數(shù)即可。不過需要快速定位某一個Catch函數(shù)。在Java中,是通過異常類型(如IOException、Exception等)定位特定的catch子句的,我們也可以模擬這一過程,通過特定的異常來定位與該異常對應(yīng)的Catch函數(shù),為了方便,可以用int類型的異常代碼。那么在調(diào)用Catch函數(shù)之前,就需要通過異常代碼先定位到某一個Catch函數(shù),然后再調(diào)用。下面就是完整的實現(xiàn)代碼。
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")
})
}
執(zhí)行結(jié)果如下圖所示。
image.png
這個實現(xiàn)與Java中的try...catch...finally的唯一區(qū)別就是必須要調(diào)用Finally函數(shù),因為處理異常的代碼都在Finally函數(shù)中。不過這并不影響使用,如果finally部分沒什么需要處理的,那么就設(shè)置一個空函數(shù)即可。
為了方便大家,我已經(jīng)將該實現(xiàn)封裝成了函數(shù)庫,調(diào)用代碼如下:
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")
})
}
感謝各位的閱讀,以上就是“如何用Go語言異常機制模擬TryCatch異常捕捉”的內(nèi)容了,經(jīng)過本文的學(xué)習(xí)后,相信大家對如何用Go語言異常機制模擬TryCatch異常捕捉這一問題有了更深刻的體會,具體使用情況還需要大家實踐驗證。這里是創(chuàng)新互聯(lián),小編將為大家推送更多相關(guān)知識點的文章,歡迎關(guān)注!
本文名稱:如何用Go語言異常機制模擬TryCatch異常捕捉
文章URL:http://aaarwkj.com/article22/gghecc.html
成都網(wǎng)站建設(shè)公司_創(chuàng)新互聯(lián),為您提供全網(wǎng)營銷推廣、靜態(tài)網(wǎng)站、響應(yīng)式網(wǎng)站、網(wǎng)站設(shè)計、、小程序開發(fā)
聲明:本網(wǎng)站發(fā)布的內(nèi)容(圖片、視頻和文字)以用戶投稿、用戶轉(zhuǎn)載內(nèi)容為主,如果涉及侵權(quán)請盡快告知,我們將會在第一時間刪除。文章觀點不代表本網(wǎng)站立場,如需處理請聯(lián)系客服。電話:028-86922220;郵箱:631063699@qq.com。內(nèi)容未經(jīng)允許不得轉(zhuǎn)載,或轉(zhuǎn)載時需注明來源: 創(chuàng)新互聯(lián)