溫馨提示×

溫馨提示×

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

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

Go中的panic/recover怎么使用

發(fā)布時間:2023-04-28 14:05:28 來源:億速云 閱讀:95 作者:iii 欄目:開發(fā)技術(shù)

本文小編為大家詳細介紹“Go中的panic/recover怎么使用”,內(nèi)容詳細,步驟清晰,細節(jié)處理妥當,希望這篇“Go中的panic/recover怎么使用”文章能幫助大家解決疑惑,下面跟著小編的思路慢慢深入,一起來學習新知識吧。

    簡介

    go語言追求簡潔,所以go語言中沒有try…catch語句。因為go語言的作者認為將異常和控制語句混在一起,很容易讓這個程序變得混亂,異常也很容易被濫用。 所以在go語言中,為了防止異常被濫用。我們常常使用函數(shù)的返回值來返回錯誤,而不是用異常來代替錯誤。如果在一些場景下確實需要處理異常,就可以使用panic和recover。panic用來拋出異常,recover用來恢復異常。

    panic是Go語言中,用于終止程序的一種函數(shù),往往用在下面兩種情況:1)程序出現(xiàn)了很大的故障,例如不能在提供服務了。2)程序在運行階段碰到了內(nèi)存異常的操作,例如空指針的取值,改寫只讀內(nèi)存等。對于panic來說,1)場景往往是主動調(diào)用;2)場景則是被動調(diào)用,panic一旦產(chǎn)生之后,會將堆棧里面的數(shù)據(jù)dump出來,這樣就方便了開發(fā)人員來定位問題。recover是用來截獲panic異常信息的,截獲了之后,可以控制程序跳過panic的地方繼續(xù)執(zhí)行。

    需要注意:

    • panic 能夠改變程序的控制流,調(diào)用 panic 后會立刻停止執(zhí)行當前函數(shù)的剩余代碼,并在當前 Goroutine 中遞歸執(zhí)行調(diào)用方的 defer;

    • recover 可以中止 panic 造成的程序崩潰。它是一個只能在 defer 中發(fā)揮作用的函數(shù),在其他作用域中調(diào)用不會發(fā)揮作用;

    1.特性

    • panic 只會觸發(fā)當前goroutine的defer

    • revoce 只有在defer中調(diào)用才能生效

    • panic 允許在defer中嵌套多磁調(diào)用

    2.panic觸發(fā)流程

    • 1.如果函數(shù)F中書寫并觸發(fā)了panic語句,會終止其后要執(zhí)行的代碼。在panic所在函數(shù)F內(nèi)如果存在要執(zhí)行的defer函數(shù)列表,則按照defer書寫順序的逆序執(zhí)行;

    • 2.如果函數(shù)G調(diào)用函數(shù)F,則函數(shù)F panic后返回調(diào)用者函數(shù)G。函數(shù)G中,調(diào)用函數(shù)F語句之后的語句都不會執(zhí)行。假如函數(shù)G中也有要執(zhí)行的defer函數(shù)列表,則按照defer書寫順序的逆序子還行;

    • 退出整個goroutine,并報告錯誤。

    3.recover使用要點

    • recover的作用是捕獲panic,從而恢復正常代碼執(zhí)行;

    • recover必須配合defer使用;

    • recover沒有傳入?yún)?shù),但是有返回值,返回值就是panic傳遞的值

    4.使用場景

    一般情況下有兩種情況用到:

    • 程序遇到無法執(zhí)行下去的錯誤時,拋出錯誤,主動結(jié)束運行。

    • 在調(diào)試程序時,通過 panic 來打印堆棧,方便定位錯誤。

    一、實踐

    1.跨線程失效

    package main
    import (
    	"fmt"
    	"time"
    )
    func main() {
    	// 主線程中的defer函數(shù)并不會執(zhí)行,因為子協(xié)程 panic后,主線程中的defer并不會執(zhí)行
    	defer println("in main")
    	go func() {
    		defer println("in goroutine")
    		fmt.Println("子協(xié)程running")
    		panic("子協(xié)程崩潰")
    	}()
    	time.Sleep(1 * time.Second)
    }
    # 輸出
    $ go run main.go
    子協(xié)程running
    in goroutine
    panic: 子協(xié)程崩潰
    goroutine 6 [running]:
    main.main.func1()

    當運行這段代碼時會發(fā)現(xiàn) main 函數(shù)中的 defer 語句并沒有執(zhí)行,執(zhí)行的只有當前 Goroutine 中的 defer。

    2.不起作用的recover

    初學 Go 語言工程師可能會寫出下面的代碼,在主程序中調(diào)用 recover 試圖中止程序的崩潰,但是從運行的結(jié)果中也能看出,下面的程序沒有正常退出。

    package main
    import "fmt"
    func main() {
    	defer fmt.Println("in main")
    	if err := recover(); err != nil {
    		fmt.Println(err)
    	}
    	panic("unknown err")
    }
    # 輸出
    $ go run main.go
    in main
    panic: unknown err
    goroutine 1 [running]:
    main.main()
            D:/gopath/src/Go_base/lesson/panic/demo5.go:11 +0x125

    仔細分析一下這個過程就能理解這種現(xiàn)象背后的原因,recover 只有在發(fā)生 panic 之后調(diào)用才會生效。然而在上面的控制流中,recover 是在 panic 之前調(diào)用的,并不滿足生效的條件,所以我們需要在 defer 中使用 recover 關(guān)鍵字。

    正確的寫法應該是這樣:

    package main
    import "fmt"
    func main() {
    	defer fmt.Println("in main")
    	defer func() {
    		if err := recover(); err != nil {
    			fmt.Println("occur error")
    			fmt.Println(err)
    		}
    	}()
    	panic("unknown err")
    }

    3.嵌套使用panic

    panic 是可以多次嵌套調(diào)用的。,如下所示的代碼就展示了如何在 defer 函數(shù)中多次調(diào)用 panic:

    package main
    import "fmt"
    func main() {
    	defer fmt.Println("in main")
    	defer func() {
    		defer func() {
    			panic("panic again and again")
    		}()
    		panic("panic again")
    	}()
    	panic("panic once")
    }
    # 輸出
    $ go run main.go
    in main
    panic: panic once
            panic: panic again
            panic: panic again and again
    goroutine 1 [running]:
    main.main.func1.1()

    從上述程序輸出的結(jié)果,我們可以確定程序多次調(diào)用 panic 也不會影響 defer 函數(shù)的正常執(zhí)行,所以使用 defer 進行收尾工作一般來說都是安全的。

    4.注意事項

    1.recover 語法

    //以下捕獲失敗
    defer recover()
    defer fmt.Prinntln(recover)
    defer func(){
        func(){
            recover() //無效,嵌套兩層
        }()
    }()
    //以下捕獲有效
    defer func(){
        recover()
    }()
    func except(){
        recover()
    }
    func test(){
        defer except()
        panic("runtime error")
    }

    2.多個panic只會捕捉最后一個

    package main
    import "fmt"
    func main(){
        defer func(){
            if err := recover() ; err != nil {
                fmt.Println(err)
            }
        }()
        defer func(){
            panic("three")
        }()
        defer func(){
            panic("two")
        }()
        panic("one")
    }

    讀到這里,這篇“Go中的panic/recover怎么使用”文章已經(jīng)介紹完畢,想要掌握這篇文章的知識點還需要大家自己動手實踐使用過才能領(lǐng)會,如果想了解更多相關(guān)內(nèi)容的文章,歡迎關(guān)注億速云行業(yè)資訊頻道。

    向AI問一下細節(jié)

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

    AI