成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

新提案:Go panic 能不能加個 PanicError?

開發 前端
在 Go 中對 panic 的優化,官方一直都是比較遲緩的。一方面是大佬們比較少寫業務代碼,另外一方面是類似對 panic 加全局攔截器避免崩潰等方式,也比較違背開創語言時的哲學宗旨。

大家好,我是煎魚。

在我們學習和平時使用 Go 時,一定會涉及到一個內置函數 panic:

func panic(v any)

調用該函數后會停止代碼的控制流程并開始恐慌,達到扭轉當前程序控制流的目的。在使用上也常常和 defer 和 recover 關聯上。

快速 Demo

以下是一個簡單的使用 Demo:

func main() {
 panic("腦子進煎魚了")

 _, err := os.Create("/tmp/file")
 if err != nil {
  log.Fatalln(err)
 }
}

輸出結果:

$ go run demo.go 
panic: 腦子進煎魚了

goroutine 1 [running]:
main.main()
 /Users/eddycjy/demo.go:10 +0x25
exit status 2

看著都沒什么問題。輸出結果符合預期。

一點爭議

由于 Go 起協程(goroutine)非常簡單、方便,因此絕大部分開發者在應用程序中會經常用 goroutine 去做各種并發處理的邏輯,一看不小心。就很有可能會引發程序中的 panic,導致整個應該程序崩潰,出現事故。(見過好幾起低級錯誤了,覺得程序沒問題,所以也沒有主動加防御性代碼)

有一個比較常見觸發的場景之一:空指針調用。時不時就能見到幾個應用又誘發了。

如下代碼:

type T struct {
 Name string
}

func main() {
 var user *T
 go func() {
  // 異步執行一些業務流程,不小心 panic 了...
  fmt.Println(user.Name)
 }()

 // 做一些事情...
 time.Sleep(time.Second * 1)

 fmt.Println("腦子進煎魚了")
}

輸出結果:

$ go run demo.go
panic: runtime error: invalid memory address or nil pointer dereference
[signal SIGSEGV: segmentation violation code=0x1 addr=0x8 pc=0x1087178]

goroutine 6 [running]:
main.main.func1()
 /Users/eddycjy/demo.go:16 +0x18
created by main.main in goroutine 1
 /Users/eddycjy/demo.go:14 +0x31
exit status 2

當然,這也是有辦法解決的。標準的方式是通過 recover,捕獲 panic。如下代碼:

go func() {
  defer func() {
   if r := recover(); r != nil {
    fmt.Println("Recovered in f", r)
   }
  }()
  // 異步執行一些業務流程,不小心 panic 了...
  fmt.Println(user.Name)
 }()

輸出結果:

Recovered in f runtime error: invalid memory address or nil pointer dereference
腦子進煎魚了

又或是基于 goroutine+recvoer 封裝一個協程調用的方法。要求使用這類工具庫來規避這個 “坑”。

但不得不說,很多同學崩就崩在不覺得這個地方會出問題,但就是有問題。最后只能一溜煙全都用封裝好的工具庫來起 goroutine 了。

新提案:可定義 panic 錯誤信息

在前面的案例中,我們可以看到 panic 后現在的輸出信息如下:

panic: runtime error: invalid memory address or nil pointer dereference
[signal SIGSEGV: segmentation violation code=0x1 addr=0x8 pc=0x1087178]

goroutine 6 [running]:
main.main.func1()
 /Users/eddycjy/demo.go:16 +0x18
created by main.main in goroutine 1
 /Users/eddycjy/demo.go:14 +0x31
exit status 2

程序輸出了恐慌值和 goroutine 堆棧跟蹤。

這第一眼看起來是非常迷惑的,要看錯誤信息。如果是程序內拋的空指針,還要去翻堆棧信息去猜,再看是哪里的程序。做一輪排查、定位、驗證。

因此社區里 @Mitar 提出了《proposal: runtime: provide a way to format output in unhandled panics[1]》的提案。希望可以針對意外情況+無人處理的 panic 錯誤進行自定義的格式化處理。

提案中希望 panic 新增 PanicError:

type panicError interface {
    error
    PanicError() string
}

如果值實現了該接口,在 panic 時則會優先調用 PanicError 方法,為錯誤處理提供一個可選選項,可以為調試補充額外的有用信息。

這樣就可以進一步區分出 Panic 錯誤和普通 Error 錯誤的方法,并且針對 Panic 的錯誤做各種奇怪的操作和補充。

總結

今天給大家分享了社區對于 panic 優化的一個小點。原提案作者的目的是為了針對 panic 錯誤新增 PanicError 方法,若存在則優先使用該方法,而非與普通 error 共用 Error 方法,并以此去做好區分識別和實現。

在 Go 中對 panic 的優化,官方一直都是比較遲緩的。一方面是大佬們比較少寫業務代碼,另外一方面是類似對 panic 加全局攔截器避免崩潰等方式,也比較違背開創語言時的哲學宗旨。

責任編輯:武曉燕 來源: 腦子進煎魚了
相關推薦

2019-11-21 09:25:23

AI 數據人工智能

2021-02-25 15:51:41

Go語言模糊測試功能

2022-07-08 08:55:56

Go函數模型

2016-05-19 17:10:27

銀行

2013-04-19 10:42:02

打車軟件大數據

2025-04-22 08:00:00

2021-02-26 21:25:08

比特幣投資貨幣

2022-10-20 08:00:37

機器人ZadigChatOps

2020-10-16 18:33:18

Rust語言前端開發

2020-12-21 15:09:23

人工智能安全人臉識別

2010-04-13 10:02:16

索引

2024-04-26 09:37:43

國產數據庫開發者

2023-04-06 06:55:24

ChatGPTGPT算力

2022-04-24 11:52:04

元宇宙Web3去中心化

2012-06-13 11:01:59

英特爾

2021-12-13 08:52:42

Go 泛型

2022-11-15 09:16:59

2011-12-06 10:06:33

云存儲

2022-11-07 08:36:11

2022-12-09 08:22:26

Gradle編譯運行
點贊
收藏

51CTO技術棧公眾號

主站蜘蛛池模板: 免费观看的黄色网址 | 澳门永久av免费网站 | 最新中文字幕久久 | 亚洲综合一区二区三区 | 一区二区av | 欧美日韩在线精品 | 久久最新| 99re国产视频| 国产伦精品一区二区三区视频金莲 | 日韩视频在线一区 | 亚洲欧美在线视频 | 国产一区二区av | 精品国产乱码一区二区三 | 看片国产 | 欧美一区二区三区高清视频 | 中文字幕一级毛片 | 一区二区三区不卡视频 | 一区二区精品 | 91成人免费 | 久热免费 | 爱草视频| 一级高清视频 | 成年人在线电影 | 99在线资源 | 亚洲欧美国产一区二区三区 | 色婷婷亚洲国产女人的天堂 | 亚洲h视频| 国产精品一区二区在线 | 久操av在线 | 成人一区二区三区在线观看 | 亚洲不卡在线视频 | 成人综合久久 | 亚洲国产成人一区二区 | 国产欧美精品一区二区色综合朱莉 | jizz亚洲人 | 久久丝袜视频 | 日韩中文字幕免费在线 | 91精品在线播放 | 精精国产xxxx视频在线野外 | 国产精品视频999 | 涩涩视频网站在线观看 |