go goroutine 怎樣進(jìn)行錯(cuò)誤處理
前言
在 Go 語(yǔ)言程序開(kāi)發(fā)中,goroutine 的使用是比較頻繁的,因此在日常編碼的時(shí)候 goroutine 里的錯(cuò)誤處理,怎么做會(huì)比較好呢?
一般我們的業(yè)務(wù)代碼如下:
func main() { var wg sync.WaitGroup wg.Add(2) go func() { //... 業(yè)務(wù)邏輯 wg.Done() }() go func() { //... 業(yè)務(wù)邏輯 wg.Done() }() wg.Wait() }
在上面的代碼中,我們運(yùn)行了多個(gè) goroutine,每個(gè)協(xié)程又是單獨(dú)行動(dòng)的,想要拋出 error 錯(cuò)誤信息,也不怎么明智。
通過(guò)錯(cuò)誤日志記錄
常用的第一種方法:通過(guò)把錯(cuò)誤記錄寫(xiě)入到日志文件中,再結(jié)合相關(guān)的 logtail 進(jìn)行采集和梳理。
但這又會(huì)引入新的問(wèn)題,那就是調(diào)用錯(cuò)誤日志的方法寫(xiě)的到處都是,代碼結(jié)構(gòu)也比較亂、不直觀。
最重要的是無(wú)法針對(duì) error 做特定的邏輯處理和流轉(zhuǎn)。
利用 channel 傳輸
大家可能會(huì)想到 Go 的經(jīng)典哲學(xué):不要通過(guò)共享內(nèi)存來(lái)通信,而是通過(guò)通信來(lái)實(shí)現(xiàn)內(nèi)存共享(Do not communicate by sharing memory; instead, share memory by communicating)。
第二種方法:利用 channel 來(lái)傳輸多個(gè) goroutine 中的 errors:
func main() { cherrors := make(chan error) wgDone := make(chan bool) var wg sync.WaitGroup wg.Add(2) go func() { //... 業(yè)務(wù)邏輯 wg.Done() }() go func() { //... 業(yè)務(wù)邏輯 err := returnErr() if err != nil { cherrors <- err } wg.Done() }() go func() { wg.Wait() close(wgDone) }() select { case <-wgDone: break case err := <-cherrors: close(cherrors) fmt.Println(err) } time.Sleep(time.Second) } func returnErr() error { return errors.New("出錯(cuò)啦。。我是錯(cuò)誤信息") }
雖然使用 channel 后已經(jīng)方便了不少,但編寫(xiě) channel 還要關(guān)心一些非業(yè)務(wù)向的邏輯。
使用 sync/errgroup
第三種方法,就是使用官方提供的 golang.org/x/sync/errgroup
標(biāo)準(zhǔn)庫(kù):
type Group func WithContext(ctx context.Context) (*Group, context.Context) func (g *Group) Go(f func() error) func (g *Group) Wait() error
- Go:?jiǎn)?dòng)一個(gè)協(xié)程,在新的 goroutine 中調(diào)用給定的函數(shù)。
- Wait:等待協(xié)程結(jié)束,直到 Go 方法中的所有函數(shù)調(diào)用都返回,然后返回其中第一個(gè)非零錯(cuò)誤(如果有錯(cuò)誤的話)。
結(jié)合其特性能夠非常便捷的針對(duì)多 goroutine 進(jìn)行錯(cuò)誤處理:
func main() { group := new(errgroup.Group) nums := []int{-1, 0, 1} for _, num := range nums { num := num group.Go(func() error { res, err := output(num) fmt.Println(res) return err }) } if err := group.Wait(); err != nil { fmt.Println("Get errors: ", err) } else { fmt.Println("Get all num successfully!") } } func output(num int) (int, error) { if num < 0 { return 0, errors.New("math: square root error!") } return num, nil }
每啟動(dòng)一個(gè)新的 goroutine 都直接使用 Group.Go
方法,在等待和錯(cuò)誤處理上使用 Group.Wait
方法。
這種方法進(jìn)行錯(cuò)誤處理的好處是不需要關(guān)注非業(yè)務(wù)邏輯的控制代碼,比較簡(jiǎn)潔明了。
總結(jié)
在 Go 語(yǔ)言中,goroutine 是一種常用的方法,為此我們需要更了解 goroutine 的一系列相關(guān)知識(shí),像是 context、error處理等
在團(tuán)隊(duì)開(kāi)發(fā)中,統(tǒng)一一定的規(guī)范,這樣的代碼閱讀起來(lái)就會(huì)比較明朗,一些隱藏很深的 Bug 也會(huì)減少很多。
到此這篇關(guān)于go goroutine 怎樣進(jìn)行錯(cuò)誤處理的文章就介紹到這了,更多相關(guān)go goroutine 錯(cuò)誤處理內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
golang協(xié)程與線程區(qū)別簡(jiǎn)要介紹
這篇文章主要介紹了golang協(xié)程與線程區(qū)別簡(jiǎn)要介紹,進(jìn)程是操作系統(tǒng)資源分配的基本單位,是程序運(yùn)行的實(shí)例,線程是操作系統(tǒng)調(diào)度到CPU中執(zhí)行的基本單位2022-06-06Go語(yǔ)言中的格式化占位符的實(shí)現(xiàn)示例
在Go語(yǔ)言中,fmt包提供了豐富的格式化占位符用于輸出不同類型的數(shù)據(jù),了解和選擇合適的占位符對(duì)于確保輸出內(nèi)容的正確性和可讀性至關(guān)重要,本文就來(lái)介紹一下,感興趣的可以學(xué)習(xí)2024-10-10基于golang的簡(jiǎn)單分布式延時(shí)隊(duì)列服務(wù)的實(shí)現(xiàn)
這篇文章主要介紹了基于golang的簡(jiǎn)單分布式延時(shí)隊(duì)列服務(wù)的實(shí)現(xiàn),文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2020-02-02Air實(shí)現(xiàn)Go程序?qū)崟r(shí)熱重載使用過(guò)程解析示例
這篇文章主要為大家介紹了Air實(shí)現(xiàn)Go程序?qū)崟r(shí)熱重載使用過(guò)程解析示例,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步早日升職加薪2022-04-04golang跳轉(zhuǎn)語(yǔ)句goto,break,continue的使用及區(qū)別說(shuō)明
這篇文章主要介紹了golang跳轉(zhuǎn)語(yǔ)句goto,break,continue的使用及區(qū)別說(shuō)明,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2020-12-12