Golang中panic與recover的區別
前言
與defer類似的是,goroutine 中也有一個_panic鏈表頭指針指向一個_panic鏈,發生panic的時候也是在鏈表頭插入_panic結構體(執行gopanic)
在執行過程中發生瞭panic。那麼panic以後的代碼不會執行,轉而執行panic的邏輯,再執行defer,執行到的defer要將started標記為true,同時將其defer結構體中的_panic指針指向當前的_panic,表示這個defer是由該panic觸發的。再去執行defer鏈表,如果defer執行中還觸發瞭panic,panic後的代碼不載執行,將這個panic插入panic鏈頭,同時將其作為當前panic。當遇到瞭與當前panic不符的defer,就找到該defer上的panic,將其標記為已終止,從defer鏈表中移除當前執行的defer。打印panic移除信息,從鏈表尾開始逐步輸出
流程
panic執行defer的流程:
- 先標記
started=true,_panic=&panic
- 後釋放
- 目的是為瞭終止之前發生的panic
異常信息的輸出方式:
- 所有還在panic鏈表上的項會被輸出
- 順序與發生panic的順序一致
// A _panic holds information about an active panic. // // A _panic value must only ever live on the stack. // // The argp and link fields are stack pointers, but don't need special // handling during stack growth: because they are pointer-typed and // _panic values only live on the stack, regular stack pointer // adjustment takes care of them. type _panic struct { // argp 存儲當前要執行的defer的函數參數地址 argp unsafe.Pointer // pointer to arguments of deferred call run during panic; cannot move - known to liblink // arg panic函數自己的參數 arg interface{} // argument to panic // link,鏈到之前發生的panic link *_panic // link to earlier panic pc uintptr // where to return to in runtime if this panic is bypassed sp unsafe.Pointer // where to return to in runtime if this panic is bypassed // recovered 標識panic是否被恢復 recovered bool // whether this panic is over // aborted 標識panic是否被終止 aborted bool // the panic was aborted goexit bool }
關於recover
recover隻執行一件事
- 將當前執行的panic的recovered字段置為true
在每個defer執行完以後panic處理流程都會檢查當前panic是否被recover
- 如果當前panic已經被恢復,就會將它從panic鏈中移除
- 執行到的defer也會被移除,同時要保存_defer.sp和_defer.pc
利用_defer.sp和_defer.pc跳出當前panic的處理流程,通過棧指針判斷,隻執行當前函數中註冊的defer函數
在發生recover的函數正常結束後才會進入到檢測panic是否被恢復的流程
當recover的函數又發生panic時,goroutine會將該panic加入到鏈頭,設置為當前panic,再去執行defer鏈表,發現當前defer是當前panic執行的,移除當前defer,繼續執行下一個,直到發現不是當前panic執行的,在panic鏈上找到那個panic,輸出異常信息
對於已經recover標記的panic在輸出異常信息時會加上recovered標記
到此這篇關於Golang中panic與recover的區別的文章就介紹到這瞭,更多相關Go panic recover內容請搜索WalkonNet以前的文章或繼續瀏覽下面的相關文章希望大傢以後多多支持WalkonNet!
推薦閱讀:
- Golang的關鍵字defer的使用方法
- Go defer 原理和源碼剖析(推薦)
- Golang中panic的異常處理
- Go錯誤和異常CGO fallthrough處理教程詳解
- golang pprof 監控goroutine thread統計原理詳解