Golang中defer與recover的組合使用示例代碼
recover運行的條件:
- 該協(xié)程必須出現(xiàn)了panic
- recover函數(shù)必須在和panic同級的defer中被調(diào)用
在Go語言中,defer和recover是兩個關(guān)鍵特性,通常結(jié)合使用以處理資源管理和異?;謴?。以下是它們的核心應用場景及使用示例:
1. defer 的應用場景
defer用于延遲執(zhí)行函數(shù)調(diào)用,確保在函數(shù)退出前執(zhí)行特定操作。主要用途包括:
資源釋放
文件操作:確保文件句柄關(guān)閉。
func readFile(filename string) error { file, err := os.Open(filename) if err != nil { return err } defer file.Close() // 確保函數(shù)返回前關(guān)閉文件 // 處理文件內(nèi)容... return nil }鎖釋放:防止死鎖。
var mu sync.Mutex func updateData() { mu.Lock() defer mu.Unlock() // 函數(shù)退出時自動釋放鎖 // 修改共享數(shù)據(jù)... }
事務(wù)回滾
- 數(shù)據(jù)庫或業(yè)務(wù)邏輯中,確保操作失敗時回滾。
func transferMoney() { tx := db.Begin() defer func() { if r := recover(); r != nil { // 結(jié)合recover處理panic tx.Rollback() } }() // 執(zhí)行轉(zhuǎn)賬操作,可能觸發(fā)panic tx.Commit() }
2. recover 的應用場景
recover用于捕獲panic,防止程序非正常終止。必須在defer函數(shù)中調(diào)用。
全局異?;謴?/h3>
- 防止因未處理的
panic導致程序崩潰。
func safeCall() {
defer func() {
if r := recover(); r != nil {
fmt.Println("Recovered from panic:", r)
}
}()
// 可能觸發(fā)panic的代碼
panic("unexpected error")
}
panic導致程序崩潰。
func safeCall() {
defer func() {
if r := recover(); r != nil {
fmt.Println("Recovered from panic:", r)
}
}()
// 可能觸發(fā)panic的代碼
panic("unexpected error")
}
保護Goroutine
- 避免某個Goroutine的
panic影響整個程序。func startWorker() { go func() { defer func() { if r := recover(); r != nil { log.Println("Worker panic:", r) } }() // Goroutine的業(yè)務(wù)邏輯... }() }
優(yōu)雅降級
- 將
panic轉(zhuǎn)換為錯誤,保持服務(wù)可用性。func safeHandler() (err error) { defer func() { if r := recover(); r != nil { err = fmt.Errorf("internal error: %v", r) } }() // 可能panic的代碼 return nil }
3. 結(jié)合使用示例
func processRequest() (err error) {
// 恢復panic并轉(zhuǎn)為錯誤
defer func() {
if r := recover(); r != nil {
err = fmt.Errorf("panic occurred: %v", r)
}
}()
// 資源管理示例
file, err := os.Open("data.txt")
if err != nil {
return err
}
defer file.Close() // 確保文件關(guān)閉
// 業(yè)務(wù)邏輯(可能觸發(fā)panic)
if someCondition {
panic("data corruption")
}
return nil
}
4. 注意事項
recover僅在defer中有效:非defer上下文中調(diào)用會返回nil。- 避免濫用
recover:隱藏panic可能導致未知狀態(tài),應僅在必要時使用。 - 明確錯誤處理:優(yōu)先返回錯誤而非依賴
panic/recover,后者適用于不可恢復的異常(如程序邏輯錯誤)。
通過合理使用defer和recover,可以顯著提升Go程序的健壯性和可維護性,尤其在資源管理和異?;謴蛨鼍爸小?/p>
在Go語言中,上述代碼無法正確捕獲panic,原因如下:
5. 問題分析
代碼示例
func safeCall() {
// 直接調(diào)用recover(不在defer中)
if r := recover(); r != nil {
fmt.Println("Recovered from panic:", r)
}
// 觸發(fā)新的panic
panic("unexpected error")
}
// 將safeCall包裹在defer中
defer func() { safeCall() }()
關(guān)鍵問題
recover未在defer中調(diào)用:
safeCall中的recover直接調(diào)用,而非通過defer注冊的函數(shù)。此時recover會在safeCall正常執(zhí)行時立即觸發(fā),而非在panic發(fā)生后被動調(diào)用。- 若此時未發(fā)生
panic,recover返回nil,無法捕獲后續(xù)觸發(fā)的panic。
panic與defer執(zhí)行順序:
- 當外層函數(shù)觸發(fā)
panic時,會先執(zhí)行已注冊的defer函數(shù)。 - 在
defer中調(diào)用safeCall,此時safeCall內(nèi)的recover嘗試捕獲當前panic,但隨后safeCall自身又觸發(fā)了一個新的panic("unexpected error"),而新的panic未被任何recover處理,導致程序崩潰。
- 當外層函數(shù)觸發(fā)
6. 正確寫法
修復方案
將recover放在defer函數(shù)中,并直接與可能觸發(fā)panic的代碼關(guān)聯(lián):
func safeCall() {
// 可能觸發(fā)panic的代碼
defer func() {
// 在defer中調(diào)用recover
if r := recover(); r != nil {
fmt.Println("Recovered from panic:", r)
}
}()
panic("unexpected error")
}
// 注冊defer
defer safeCall()
執(zhí)行邏輯
- 調(diào)用
defer safeCall(),注冊safeCall到外層函數(shù)的defer棧。 - 當外層函數(shù)觸發(fā)
panic時,執(zhí)行safeCall。 safeCall內(nèi)部的defer函數(shù)中的recover會捕獲當前panic,阻止其繼續(xù)傳播。- 若
safeCall自身觸發(fā)panic,該panic會被其自身的defer recover捕獲。
7. 錯誤示例的詳細解釋
原代碼執(zhí)行流程
假設(shè)外層函數(shù)觸發(fā)panic:
- 外層函數(shù)執(zhí)行
panic("outer panic")。 - 程序開始處理
defer,調(diào)用defer func() { safeCall() }()。 safeCall執(zhí)行:recover()嘗試捕獲外層panic("outer panic"),打印恢復信息。- 隨后觸發(fā)新的
panic("unexpected error")。
- 新的panic未被任何recover處理,導致程序崩潰。
關(guān)鍵結(jié)論
recover必須通過defer注冊的函數(shù)被動調(diào)用,才能捕獲到panic。- 若在普通代碼中直接調(diào)用
recover,只有在已發(fā)生panic且未被處理時才會生效。
總結(jié)
- 必須將recover放在defer函數(shù)中,才能確保在
panic發(fā)生后被動調(diào)用。 - 避免在恢復邏輯中觸發(fā)新的panic,否則需要額外的
recover處理。 - 正確的
defer和recover組合是資源管理和異?;謴偷暮诵哪J健?/li>
通過調(diào)整代碼結(jié)構(gòu),確保recover在defer中調(diào)用,即可正確捕獲并處理panic。
到此這篇關(guān)于Golang中defer與recover組合使用的文章就介紹到這了,更多相關(guān)Go defer與recover組合使用內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
goland 實現(xiàn)websocket server的示例代碼
本文主要介紹了goland 實現(xiàn)websocket server的示例代碼,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧2022-06-06
GO 使用Webhook 實現(xiàn)github 自動化部署的方法
這篇文章主要介紹了GO 使用Webhook 實現(xiàn)github 自動化部署的方法,本文通過實例代碼給大家介紹的非常詳細,對大家的學習或工作具有一定的參考借鑒價值,需要的朋友可以參考下2020-05-05

