溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點(diǎn)擊 登錄注冊 即表示同意《億速云用戶服務(wù)條款》

golang?recover函數(shù)使用的坑怎么解決

發(fā)布時(shí)間:2023-02-24 16:28:09 來源:億速云 閱讀:210 作者:iii 欄目:開發(fā)技術(shù)

這篇文章主要講解了“golang recover函數(shù)使用的坑怎么解決”,文中的講解內(nèi)容簡單清晰,易于學(xué)習(xí)與理解,下面請大家跟著小編的思路慢慢深入,一起來研究和學(xué)習(xí)“golang recover函數(shù)使用的坑怎么解決”吧!

一,正常情況下

package main
import "fmt"
func main(){
    defer func(){
        if err := recover();err != nil{
            fmt.Printf("err = %v",err)
        }
    }()
    panic("a panic")
}
打印結(jié)果:
err = a panic
Process finished with exit code 0

能正常catch panic

二, goroutine中panic 

之前線上環(huán)境出現(xiàn)過接口出現(xiàn)panic導(dǎo)致服務(wù)不可用的情況,于是同事就直接在main函數(shù)加了個(gè)recover認(rèn)為萬事無憂了。實(shí)際上recover并不能捕捉到協(xié)程中的panic。

package main
import "fmt"
func main(){
    defer func(){
        if err := recover();err != nil{
            fmt.Printf("err = %v",err)
        }
    }()
    go func(){
        panic("a panic")
    }()
    select{}
}
打印結(jié)果:
panic: a panic
goroutine 6 [running]:
main.main.func2()
    I:/goProject/catchPanic.go:13 +0x40
created by main.main
    I:/goProject/catchPanic.go:12 +0x5e

實(shí)際上還是會(huì)panic導(dǎo)致服務(wù)不可用。

正確寫法

package main
import "fmt"
func main(){
    go func(){
        defer func(){
            if err := recover();err != nil{
                fmt.Printf("err = %v",err)
            }
        }()
        panic("a panic")
    }()
    select {}
}
返回值:
fatal error: all goroutines are asleep - deadlock!
goroutine 1 [select (no cases)]:
main.main()
    I:/goProject/catchPanic.go:15 +0x41
err = a panic
Process finished with exit code 2

可以看到panic被正常捕捉,同時(shí)因?yàn)閟elect語句陷入阻塞,報(bào)了一個(gè)死鎖的錯(cuò)。

三,間接調(diào)用recover

在我想要把recover封裝成成一個(gè)函數(shù)的時(shí)候,發(fā)現(xiàn)recover并沒有生效,因?yàn)閞ecover只有在被defer語句直接調(diào)用的時(shí)候才會(huì)生效。當(dāng)recover在其他函數(shù)內(nèi)部的時(shí)候無法正確捕捉到panic。

package main
import "fmt"
func main(){
    defer cover()
    panic("a panic")
}
func cover(){
    defer func(){
        if err := recover();err!= nil{
            fmt.Println(err)
        }
    }()
}
返回值:
panic: a panic
goroutine 1 [running]:
main.main()
    I:/goProject/catchPanic.go:7 +0x62

四,nil panic

panic要被捕捉,還需要滿足一種條件,就是panic不是nil panic,否則在進(jìn)行捕獲判斷的時(shí)候無法知道是panic沒有發(fā)生還是panic本身就是nil。

例如以下代碼

package main
import "fmt"
func main() {
    defer func(){
        if err := recover();err != nil{
            fmt.Println(err)
        }
        fmt.Println("after recover")
    }()
    panic(nil)
    select{}
}
返回值:
after recover

recover并沒有正確處理異常,因?yàn)楫惓5闹禐閚il。

感謝各位的閱讀,以上就是“golang recover函數(shù)使用的坑怎么解決”的內(nèi)容了,經(jīng)過本文的學(xué)習(xí)后,相信大家對golang recover函數(shù)使用的坑怎么解決這一問題有了更深刻的體會(huì),具體使用情況還需要大家實(shí)踐驗(yàn)證。這里是億速云,小編將為大家推送更多相關(guān)知識(shí)點(diǎn)的文章,歡迎關(guān)注!

向AI問一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場,如果涉及侵權(quán)請聯(lián)系站長郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI