GoLang中panic與recover函數(shù)以及defer語句超詳細(xì)講解
一、運行時恐慌panic
panic是一種在運行時拋出來的異常。比如"index of range"。
panic的詳情:
package main import "fmt" func main() { oneC := []int{1, 2, 3, 4, 5} v5 := oneC[5] fmt.Println(v5) }
$ go run demo01.go
panic: runtime error: index out of range [5] with length 5
goroutine 1 [running]:
main.main()
/Users/lifei/Documents/workspace/githubRepositoies/gowp/projects/go-core-example/src/article19/q1/demo01.go:7 +0x1b
exit status 2
$
打印信息的第一行,"panic: "右邊的內(nèi)容,正是panic包含的runtime.Error
類型值的字符串表示形式;
“goroutine 1 [running]” 表示有一個id為 1 的 goroutine在此panic被引發(fā)的時候正在運行;
這里的ID編號并不重要,是GO語言運行時系統(tǒng)內(nèi)部給予的一個goroutine編號,我們在程序中無法獲取,也無法改變。
再下面是指出哪一行發(fā)生錯誤。“+0x1b”代表 此行代碼相對于其所屬函數(shù)的入口程序計數(shù)偏移量, 一般用途不大。
最后的 “exit status 2”,表明我的這個程序是以退出狀態(tài)碼2結(jié)束運行的。
在大多操作系統(tǒng)中,只要退出狀態(tài)碼不是0,都意味著程序運行的非正常結(jié)束。
二、panic被引發(fā)到程序終止經(jīng)歷的過程
某個函數(shù)無疑觸發(fā)了panic:
- 初始的panic詳情會被建立起來,此行代碼所屬函數(shù)的執(zhí)行隨機終止。
- 控制權(quán)立刻轉(zhuǎn)移到上一級;
- 控制權(quán)如此一層層沿著調(diào)用棧的反方向傳播至頂端,也就是我們編寫的最外層函數(shù);
- 最終,控制權(quán)被GO語言運行時系統(tǒng)收回。隨后程序崩潰并終止運行;
panic 詳情會在控制權(quán)傳播的過程中,被逐漸地積累和完善,并且,控制權(quán)會一級一級地沿著調(diào)用棧的反方向傳播至頂端。因此,在針對某個 goroutine 的代碼執(zhí)行信息中,調(diào)用棧底端的信息會先出現(xiàn),然后是上一級調(diào)用的信息,以此類推,最后才是此調(diào)用棧頂端的信息。
三、有意引發(fā)一個panic并讓panic包含一個值
- 可以使用panic函數(shù)有意地引發(fā)一個 panic。
- 在調(diào)用panic函數(shù)時,把某個值作為參數(shù)傳給該函數(shù)就可以了。由于panic函數(shù)的唯一一個參數(shù)是空接口(也就是interface{})類型的,所以從語法上講,它可以接受任何類型的值。
- 但是,我們最好傳入error類型的錯誤值,或者其他的可以被有效序列化的值。這里的“有效序列化”指的是,可以更易讀地去表示形式轉(zhuǎn)換。
打印錯誤信息:
- 對于fmt包下的各種打印函數(shù)來說,error類型值的Error方法與其他類型值的String方法是等價的,它們的唯一結(jié)果都是string類型的;
- 如果某個值有可能會被記到日志里,那么就應(yīng)該為它關(guān)聯(lián)String方法。
四、施加應(yīng)對panic的保護(hù)措施從而避免程序崩潰
聯(lián)用defer語句和recover函數(shù)調(diào)用,才能夠恢復(fù)一個已經(jīng)發(fā)生的 panic。
GO語言的內(nèi)建函數(shù)recover專門用于恢復(fù)panic。recover函數(shù)無需任何參數(shù),并且會返回一個空接口類型的值。
defer 語句用來延遲執(zhí)行代碼。延遲到該語句所在的函數(shù)即將執(zhí)行結(jié)束的那一刻,無論結(jié)束執(zhí)行的原因是什么。
限制:有一些調(diào)用表達(dá)式是不能出現(xiàn)在這里的,包括:針對 Go 語言內(nèi)建函數(shù)的調(diào)用表達(dá)式,以及針對unsafe包中的函數(shù)的調(diào)用表達(dá)式。
package main import ( "errors" "fmt" ) func main() { fmt.Println("Enter function main") // 延遲func函數(shù)的執(zhí)行,直到main結(jié)束 defer func() { fmt.Println("Enter defer function") if p := recover(); p != nil { fmt.Printf("%v\n", p) } fmt.Println("Exit defer function") }() // 引發(fā)painc panic(errors.New("soming wrong")) fmt.Println("Exit function main") }
五、多條defer語句多條defer語句的執(zhí)行順序
在同一個函數(shù)中,defer函數(shù)調(diào)用的執(zhí)行順序與它們分別所屬的defer語句的出現(xiàn)順序(更嚴(yán)謹(jǐn)?shù)卣f,是執(zhí)行順序)完全相反。
當(dāng)一個函數(shù)即將結(jié)束執(zhí)行時,其中的寫在最下邊的defer函數(shù)調(diào)用會最先執(zhí)行,其次是寫在它上邊、與它的距離最近的那個defer函數(shù)調(diào)用,以此類推,最上邊的defer函數(shù)調(diào)用會最后一個執(zhí)行。
defer語句執(zhí)行的內(nèi)幕:
在defer語句每次執(zhí)行的時候,Go 語言會把它攜帶的defer函數(shù)及其參數(shù)值另行存儲到一個鏈表中。
這個鏈表與該defer語句所屬的函數(shù)是對應(yīng)的,并且,它是先進(jìn)后出(FILO)的,相當(dāng)于一個棧。
在需要執(zhí)行某個函數(shù)中的defer函數(shù)調(diào)用的時候,Go 語言會先拿到對應(yīng)的鏈表,然后從該鏈表中一個一個地取出defer函數(shù)及其參數(shù)值,并逐個執(zhí)行調(diào)用。
package main import "fmt" func main() { defer fmt.Println("first defer") for i := 0; i < 3; i++ { defer fmt.Printf("defer in for %d\n", i) } defer fmt.Println("last defer") }
到此這篇關(guān)于GoLang中panic與recover函數(shù)以及defer語句超詳細(xì)講解的文章就介紹到這了,更多相關(guān)Go panic recover defer內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Go語言普通指針unsafe.Pointer?uintpt之間的關(guān)系及指針運算
這篇文章主要為大家介紹了Go語言普通指針unsafe.Pointer?uintpt之間的關(guān)系及指針運算示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-12-12go?zero微服務(wù)實戰(zhàn)處理每秒上萬次的下單請求
這篇文章主要為大家介紹了go?zero微服務(wù)實戰(zhàn)處理每秒上萬次的下單請求示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2022-07-07golang利用redis和gin實現(xiàn)保存登錄狀態(tài)校驗登錄功能
這篇文章主要介紹了golang利用redis和gin實現(xiàn)保存登錄狀態(tài)校驗登錄功能,本文給大家介紹的非常詳細(xì),對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友參考下吧2024-01-01Go語言中?Print?Printf和Println?的區(qū)別解析
這篇文章主要介紹了Go語言中?Print?Printf和Println?的區(qū)別,本文給大家介紹的非常詳細(xì),對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友可以參考下2023-03-03go-zero讀取請求體出現(xiàn)EOF錯誤的解決方法
這篇文章主要為大家詳細(xì)介紹了go-zero讀取請求體出現(xiàn)EOF錯誤時如何解決,文中的示例代碼講解詳細(xì),感興趣的小伙伴可以跟隨小編一起學(xué)習(xí)一下2025-02-02