亚洲乱码中文字幕综合,中国熟女仑乱hd,亚洲精品乱拍国产一区二区三区,一本大道卡一卡二卡三乱码全集资源,又粗又黄又硬又爽的免费视频

Golang中panic的實現(xiàn)示例

 更新時間:2025年06月23日 10:51:57   作者:CarlosKeFeng  
這篇文章主要介紹了Golang中panic的實現(xiàn)示例,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧

前言

Golang中當(dāng)程序發(fā)生致命異常時(比如數(shù)組下標(biāo)越界,注意這里的異常并不是error),Golang程序會panic(運行時恐慌)。當(dāng)程序發(fā)生panic時,程序會執(zhí)行當(dāng)前棧中的defer 函數(shù)列表。然后打印引發(fā)panic的具體信息,最后進(jìn)程退出,本篇文章我們一起探討Golang中的panic以及如何利用defer 和 recover 來恢復(fù)這種致命的異常

分析造成panic堆棧信息

func main() {
	f1()
	fmt.Println("main func end")
}

func f1() {
	fmt.Println("func f1 start")
	arr := []int{}
	fmt.Println(arr[10])
	fmt.Println("func f1 end")
}

上述代碼中,我在main函數(shù)(主協(xié)程)中調(diào)用了f1函數(shù),在調(diào)用完該函數(shù)后,我打印了「main func end」,程序如果正常執(zhí)行的話會輸出

func f1 start
func f1 end
main func end

很明顯我們可以看出 f1 函數(shù)中,切片arr是沒有索引為10的元素的,這個時候程序運行時會造成panic,下面是程序panic時,console打印的堆棧信息

func f1 start
panic: runtime error: index out of range [10] with length 0
goroutine 1 [running]:
main.f1()
/Users/carlos/go/src/test/demo01.go:15 +0x78
main.main()
/Users/carlos/go/src/test/demo01.go:8 +0x20
Process finished with the exit code 2

我們從堆棧中可以發(fā)現(xiàn):

  • 程序會在造成panic所處的位置終止

我們可以看到錯誤信息中只輸出了 func f1 start

  • 產(chǎn)生panic的原因

panic: runtime error: index out of range [10] with length 0

  • 是哪里造成的panic

goroutine 1 [running] // 運行該程序的協(xié)程
main.f1()
/Users/carlos/go/src/test/demo01.go:15 +0x78 // f1 函數(shù),當(dāng)前demo01文件的低15行
main.main()
/Users/carlos/go/src/test/demo01.go:8 +0x20 // main 函數(shù),當(dāng)前文件的弟8行

從上面的panic詳情我們可以看出,錯誤鏈?zhǔn)峭ㄟ^棧的形式展現(xiàn)出來的(mian函數(shù)先調(diào)用,然后在mian中調(diào)用f1),所以大家以后在程序發(fā)生panic時查看堆棧信息時可以先看最上層的錯誤,因為這里是造成panic的根本原因

如何恢復(fù)panic造成的程序崩潰

Golang中提供了recover函數(shù)用來恢復(fù)因panic造成的程序崩潰。recover函數(shù)有一個返回值來告訴我們panic產(chǎn)生的具體原因。下面我們通過代碼來進(jìn)行演示

func main() {
	f1()
	r := recover()
	fmt.Printf("%s \n", r)
	fmt.Println("main func end")
}

func f1() {
	fmt.Println("func f1 start")
	arr := []int{}
	fmt.Println(arr[10])
	fmt.Println("func f1 end")
}

上述代碼中我只是在調(diào)用f1函數(shù)的下一行調(diào)用了recover函數(shù),這樣一來我們的理想狀態(tài)了能夠恢復(fù)程序,讓程序執(zhí)行完main函數(shù)中剩下的代碼(打印panic信息,最后打印 main func end),當(dāng)我們運行該程序的時候發(fā)現(xiàn)recover并沒有起到作用,這是因為當(dāng)f1造成panic時,f1下方的recover函數(shù)根本沒有機(jī)會執(zhí)行。
下面我將上述代碼進(jìn)行一個簡單的改造:

func main() {
	defer func() {
		fmt.Println("defer func start")
		if r := recover(); r != nil {
			fmt.Printf("%s \n", r)
		}
		fmt.Println("defer func end")
	}()
	f1()
	fmt.Println("main func end")
}

func f1() {
	fmt.Println("func f1 start")
	arr := []int{}
	fmt.Println(arr[10])
	fmt.Println("func f1 end")
}

輸出

func f1 start
defer func start
runtime error: index out of range [10] with length 0 
defer func end

上述代碼中,我只是在main函數(shù)最開頭添加了一個defer 函數(shù),并在該函數(shù)中調(diào)用了recover函數(shù)。注意,我們在文章的最開頭已經(jīng)說明了,當(dāng)程序發(fā)生panic時,程序會依次執(zhí)行棧中的defer函數(shù)(關(guān)于defer函數(shù)請閱讀官網(wǎng)描述)。所以當(dāng)前程序發(fā)生panic時在進(jìn)程退出之前會走到defer函數(shù)中執(zhí)行recover函數(shù),recover函數(shù)會恢復(fù)當(dāng)前進(jìn)程并打印錯誤信息。

這里我需要特別提醒你一點,最好將defer語句寫在函數(shù)的最前面。如果上述例子我將f1的調(diào)用寫在defer函數(shù)之前,你會發(fā)現(xiàn)recover函數(shù)還是沒有執(zhí)行

func main() {
	f1()
	defer func() {
		fmt.Println("defer func start")
		if r := recover(); r != nil {
			fmt.Printf("%s \n", r)
		}
		fmt.Println("defer func end")
	}()
	fmt.Println("main func end")
}

這是因為f1造成panic時,defer函數(shù)根本就沒有壓入函數(shù)調(diào)用棧中。

何時使用panic

當(dāng)你的項目中特別依賴一些組件時,比如一些web項目中經(jīng)常會在進(jìn)程啟動之前初始化一些mysql,mq句柄。這些實例對業(yè)務(wù)來說是非常重要的,所以當(dāng)這些實例初始化失敗時我們可以直接讓當(dāng)前程序panic(手動panic),然后及時發(fā)現(xiàn)問題并解決。這樣總比你帶著問題上線后,然后一批流入打入進(jìn)來,客戶端瘋狂報錯要好

Golang中手動調(diào)用panic:

func main() {
	initMysql()
}

func initMysql() {
	panic("init mysql failed") // panic可以接收一個interface類型的參數(shù)
}

到此這篇關(guān)于Golang中panic的實現(xiàn)示例的文章就介紹到這了,更多相關(guān)Golang panic內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家! 

相關(guān)文章

  • 詳解Golang并發(fā)控制的三種方案

    詳解Golang并發(fā)控制的三種方案

    本文主要介紹了詳解Golang并發(fā)控制的三種方案,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2024-06-06
  • 從基礎(chǔ)到高階解析Go語言中數(shù)組的應(yīng)用

    從基礎(chǔ)到高階解析Go語言中數(shù)組的應(yīng)用

    在本文中,我們將從基礎(chǔ)概念、常規(guī)操作,到高級技巧和特殊操作,帶大家深入了解Go語言中數(shù)組的各個方面,感興趣的小伙伴可以跟隨小編一起學(xué)習(xí)一下
    2023-10-10
  • Go語言實現(xiàn)順序存儲的線性表實例

    Go語言實現(xiàn)順序存儲的線性表實例

    這篇文章主要介紹了Go語言實現(xiàn)順序存儲的線性表的方法,實例分析了Go語言實現(xiàn)線性表的定義、插入、刪除元素等的使用技巧,具有一定參考借鑒價值,需要的朋友可以參考下
    2015-03-03
  • 使用Go語言與MQTT進(jìn)行通信的示例代碼

    使用Go語言與MQTT進(jìn)行通信的示例代碼

    本文介紹了如何使用 Go 編程語言與 MQTT(Message Queuing Telemetry Transport)進(jìn)行通信,MQTT 是一種輕量級的消息傳輸協(xié)議,廣泛應(yīng)用于物聯(lián)網(wǎng)和實時通信場景,通過本文的指導(dǎo),您將學(xué)習(xí)如何使用 Go 語言創(chuàng)建 MQTT 客戶端,進(jìn)行消息的發(fā)布和訂閱,需要的朋友可以參考下
    2023-12-12
  • 通過示例深度理解Go channel range

    通過示例深度理解Go channel range

    這篇文章主要為大家介紹了Go channel range使用示例深度理解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪
    2023-05-05
  • Go語言中處理JSON字符串轉(zhuǎn)換的常見方式

    Go語言中處理JSON字符串轉(zhuǎn)換的常見方式

    這篇文章主要介紹了Go語言中處理JSON字符串轉(zhuǎn)換的常見方法,這些方法覆蓋了 Go 語言中處理 JSON 字符串的常見場景,你可以根據(jù)具體需求選擇合適的方式,需要的朋友可以參考下
    2025-09-09
  • Go語言實現(xiàn)二進(jìn)制與十進(jìn)制互轉(zhuǎn)的示例代碼

    Go語言實現(xiàn)二進(jìn)制與十進(jìn)制互轉(zhuǎn)的示例代碼

    這篇文章主要和大家詳細(xì)介紹了Go語言中實現(xiàn)二進(jìn)制與十進(jìn)制互相轉(zhuǎn)換的示例代碼,文中的代碼簡潔易懂,感興趣的小伙伴可以跟隨小編一起學(xué)習(xí)一下
    2023-05-05
  • Go應(yīng)該如何實現(xiàn)二級緩存

    Go應(yīng)該如何實現(xiàn)二級緩存

    本文主要介紹了Go二級緩存,文中通過示例代碼介紹的非常詳細(xì),具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2021-08-08
  • 使用 go 實現(xiàn)多線程下載器的方法

    使用 go 實現(xiàn)多線程下載器的方法

    本篇文章帶領(lǐng)大家學(xué)習(xí)使用go實現(xiàn)一個簡單的多線程下載器,給她家詳細(xì)介紹了多線程下載原理及實例代碼,感興趣的朋友跟隨小編一起看看吧
    2021-10-10
  • Golang單元測試中的技巧分享

    Golang單元測試中的技巧分享

    這篇文章主要為大家詳細(xì)介紹了Golang進(jìn)行單元測試時的一些技巧和科技,文中的示例代碼講解詳細(xì),具有一定的參考價值,感興趣的小伙伴可以了解一下
    2023-03-03

最新評論