Go中的錯誤和異常處理最佳實踐方法
錯誤
認(rèn)識錯誤
在Go中,錯誤是一種表示程序錯誤狀態(tài)。包含了在程序在運行時、編譯時的狀態(tài)信息。一般我們在編寫Go代碼中,都會碰到如下的處理方式。
file, err := os.Create("test.txt") fmt.Println(file) if err != nil { fmt.Println(err) return }
我們使用os庫創(chuàng)建一個名為test.txt的文件,該方法返回一個文件指針或err的錯誤信息。
err表示文件創(chuàng)建失敗時的錯誤信息。當(dāng)存儲錯誤時,我們則對程序做錯誤處理;不存在錯誤時,則正常執(zhí)行其他的邏輯代碼。
自定義錯誤
在Go中是允許我們自定義錯誤信息的。自定義錯誤信息需要利用自帶的error報中的New()函數(shù)。
如下示例代碼:
package main import ( "errors" "fmt" ) func printError() (a int, err error) { err = errors.New("打印錯誤信息") a = 1 return } func main() { i, err := printError() fmt.Println("i value is", i) if err != nil { fmt.Println(err) return } }
具體的打印信息:??i value is 1 打印錯誤信息?
?。
實現(xiàn)原理
在使用??errors.New()?
?函數(shù)時,該包中聲明了一個結(jié)構(gòu)體??errorString?
?并且實現(xiàn)了error接口體中的方法??Error()?
?。
// errors包 package errors func New(text string) error { return &errorString{text} } type errorString struct { s string } func (e *errorString) Error() string { return e.s }
// error接口 type error interface { Error() string }
異常
認(rèn)識異常
異常是程序在??編譯時?
?或者??運行時?
?發(fā)生的異常信息。如果不對異常做處理,可能導(dǎo)致程序終止程序或者拋出異常信息,導(dǎo)致程序無法正常運行。不管是在程序編譯或者運行時,都需要對異常進行嚴(yán)格處理。
如下代碼,程序在編譯時就會觸發(fā)異常,導(dǎo)致無法進行正常編譯:
package main import "fmt" func main() { panic("print panic") fmt.Println("end") }
打印結(jié)果:
╰─ go run demo6.go
panic: print panic
goroutine 1 [running]:
main.main()
/usr/local/var/www/go/golang_code/src/syntax/err/demo6.go:20 +0x39
exit status 2
- Go 運行時會觸發(fā)運行時 panic,伴隨著程序的崩潰拋出一個 runtime.Error 接口類型的值。這個錯誤值有個 RuntimeError() 方法用于區(qū)別普通錯誤。
- panic 可以直接從代碼初始化:當(dāng)錯誤條件(我們所測試的代碼)很嚴(yán)苛且不可恢復(fù),程序不能繼續(xù)運行時,可以使用 panic 函數(shù)產(chǎn)生一個中止程序的運行時錯誤。
- panic 接收一個做任意類型的參數(shù),通常是字符串,在程序死亡時被打印出來。Go 運行時負(fù)責(zé)中止程序并給出調(diào)試信息。
- 在多層嵌套的函數(shù)調(diào)用中調(diào)用 panic,可以馬上中止當(dāng)前函數(shù)的執(zhí)行。
處理異常
當(dāng)程序在運行過程中發(fā)生異常,會終止程序的正常運行。需要嚴(yán)格處理異常信息。Go中可以使用recover()將程序從panic中獲取異常信息,并獲取程序的執(zhí)行權(quán)。
- 正如名字一樣,這個(recover)內(nèi)建函數(shù)被用于從 panic 或 錯誤場景中恢復(fù):讓程序可以從 panicking 重新獲得控制權(quán),停止終止過程進而恢復(fù)正常執(zhí)行。
- recover
只能
在defer修飾的函數(shù)中使用:用于取得panic調(diào)用中傳遞過來的錯誤值,如果是正常執(zhí)行,調(diào)用recover會返回 nil,且沒有其它效果。 - panic 會導(dǎo)致棧被展開直到defer修飾的recover()被調(diào)用或者程序中止。
- 所有的 defer 語句都會保證執(zhí)行并把控制權(quán)交還給接收到 panic 的函數(shù)調(diào)用者。這樣向上冒泡直到最頂層,并執(zhí)行(每層的) defer,在棧頂處程序崩潰,并在命令行中用傳給 panic 的值報告錯誤情況:這個終止過程就是 panicking。
異常處理原則
- 在包內(nèi)部,應(yīng)該從panic中recover:不允許顯式的超出包范圍的panic()。在包內(nèi)部,特別是在非導(dǎo)出函數(shù)中有很深層次的嵌套調(diào)用時,對主調(diào)函數(shù)來說用 panic 來表示應(yīng)該被翻譯成錯誤的錯誤場景是很有用的(并且提高了代碼可讀性)。
- 在包外部,向包的調(diào)用者返回錯誤值(而不是panic)。
- Go 庫的原則是即使在包的內(nèi)部使用了 panic,在它的對外接口(API)中也必須用 recover 處理成返回顯式的錯誤。
異常處理實踐
下面的示例代碼,在被調(diào)用函數(shù)printPanic()中觸發(fā)一個panic(),在main()函數(shù)中使用defer中接收panic()信息,并對panic()做異常處理。
package main import "fmt" func printPanic() { panic("panic exception") } func main() { defer func() { err := recover() if err != nil { fmt.Println("panic is", err) } }() printPanic() fmt.Println("end") }
打印結(jié)果:
╰─ go run demo5.go
i value is 1
打印錯誤信息
到此這篇關(guān)于Go中的錯誤和異常處理最佳實踐方法的文章就介紹到這了,更多相關(guān)Go異常處理內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
關(guān)于golang中map使用的幾點注意事項總結(jié)(強烈推薦!)
map是一種無序的基于key-value的數(shù)據(jù)結(jié)構(gòu),Go語言中的map是引用類型,必須初始化才能使用,下面這篇文章主要給大家介紹了關(guān)于golang中map使用的幾點注意事項,需要的朋友可以參考下2023-01-01Go條件控制語句詳解(if-else、switch和select)
條件語句用于檢查一個條件是否為真,并根據(jù)條件的真假來決定是否執(zhí)行相應(yīng)的代碼,下面這篇文章主要給大家介紹了關(guān)于Go條件控制語句(if-else、switch和select)的相關(guān)資料,需要的朋友可以參考下2024-03-03詳解如何使用Golang操作MongoDB數(shù)據(jù)庫
在現(xiàn)代開發(fā)中,數(shù)據(jù)存儲是一個至關(guān)重要的環(huán)節(jié),MongoDB作為一種NoSQL數(shù)據(jù)庫,提供了強大的功能和靈活的數(shù)據(jù)模型,與Golang的高性能和并發(fā)性能非常契合,本文將探討Golang與MongoDB的完美組合,介紹如何使用Golang操作MongoDB數(shù)據(jù)庫,需要的朋友可以參考下2023-11-11go微服務(wù)PolarisMesh源碼解析服務(wù)端啟動流程
這篇文章主要為大家介紹了go微服務(wù)PolarisMesh源碼解析服務(wù)端啟動流程詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪2023-01-01