Go基礎(chǔ)教程系列之WaitGroup用法實(shí)例詳解
正常情況下,新激活的goroutine(協(xié)程)的結(jié)束過(guò)程是不可控制的,唯一可以保證終止goroutine(協(xié)程)的行為是main goroutine(協(xié)程)的終止。也就是說(shuō),我們并不知道哪個(gè)goroutine(協(xié)程)什么時(shí)候結(jié)束。
但很多情況下,我們正需要知道goroutine(協(xié)程)是否完成。這需要借助sync包的WaitGroup來(lái)實(shí)現(xiàn)。
WatiGroup是sync包中的一個(gè)struct類(lèi)型,用來(lái)收集需要等待執(zhí)行完成的goroutine(協(xié)程)。下面是它的定義:
type WaitGroup struct { // Has unexported fields. } A WaitGroup waits for a collection of goroutines to finish. The main goroutine calls Add to set the number of goroutines to wait for. Then each of the goroutines runs and calls Done when finished. At the same time, Wait can be used to block until all goroutines have finished. A WaitGroup must not be copied after first use. func (wg *WaitGroup) Add(delta int) func (wg *WaitGroup) Done() func (wg *WaitGroup) Wait()
它有3個(gè)方法:
- Add():每次激活想要被等待完成的goroutine(協(xié)程)之前,先調(diào)用Add(),用來(lái)設(shè)置或添加要等待完成的goroutine(協(xié)程)數(shù)量
- 例如Add(2)或者兩次調(diào)用Add(1)都會(huì)設(shè)置等待計(jì)數(shù)器的值為2,表示要等待2個(gè)goroutine(協(xié)程)完成
- Done():每次需要等待的goroutine(協(xié)程)在真正完成之前,應(yīng)該調(diào)用該方法來(lái)人為表示goroutine(協(xié)程)完成了,該方法會(huì)對(duì)等待計(jì)數(shù)器減1
- Wait():在等待計(jì)數(shù)器減為0之前,Wait()會(huì)一直阻塞當(dāng)前的goroutine(協(xié)程)
也就是說(shuō),Add()用來(lái)增加要等待的goroutine(協(xié)程)的數(shù)量,Done()用來(lái)表示goroutine(協(xié)程)已經(jīng)完成了,減少一次計(jì)數(shù)器,Wait()用來(lái)等待所有需要等待的goroutine(協(xié)程)完成。
下面是一個(gè)示例,通過(guò)示例很容易理解。
package main import ( "fmt" "sync" "time" ) func process(i int, wg *sync.WaitGroup) { fmt.Println("started Goroutine ", i) time.Sleep(2 * time.Second) fmt.Printf("Goroutine %d ended\n", i) wg.Done() } func main() { no := 3 var wg sync.WaitGroup for i := 0; i < no; i++ { wg.Add(1) go process(i, &wg) } wg.Wait() fmt.Println("All go routines finished executing") }
上面激活了3個(gè)goroutine,每次激活goroutine之前,都先調(diào)用Add()方法增加一個(gè)需要等待的goroutine計(jì)數(shù)。每個(gè)goroutine都運(yùn)行process()函數(shù),這個(gè)函數(shù)在執(zhí)行完成時(shí)需要調(diào)用Done()方法來(lái)表示goroutine的結(jié)束。激活3個(gè)goroutine后,main goroutine會(huì)執(zhí)行到Wait(),由于每個(gè)激活的goroutine運(yùn)行的process()都需要睡眠2秒,所以main goroutine在Wait()這里會(huì)阻塞一段時(shí)間(大約2秒),當(dāng)所有g(shù)oroutine都完成后,等待計(jì)數(shù)器減為0,Wait()將不再阻塞,于是main goroutine得以執(zhí)行后面的Println()。
還有一點(diǎn)需要特別注意的是process()中使用指針類(lèi)型的*sync.WaitGroup
作為參數(shù),這里不能使用值類(lèi)型的sync.WaitGroup
作為參數(shù),因?yàn)檫@意味著每個(gè)goroutine都拷貝一份wg,每個(gè)goroutine都使用自己的wg。這顯然是不合理的,這3個(gè)goroutine應(yīng)該共享一個(gè)wg,才能知道這3個(gè)goroutine都完成了。實(shí)際上,如果使用值類(lèi)型的參數(shù),main goroutine將會(huì)永久阻塞而導(dǎo)致產(chǎn)生死鎖。
更多關(guān)于Go WaitGroup用法教程請(qǐng)查看下面的相關(guān)鏈接
相關(guān)文章
Goland激活碼破解永久版及安裝詳細(xì)教程(親測(cè)可以)
這篇文章主要介紹了Goland激活碼破解永久版及安裝詳細(xì)教程(親測(cè)可以),本文通過(guò)實(shí)例圖文相結(jié)合給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2020-10-10Go語(yǔ)言學(xué)習(xí)之接口類(lèi)型(interface)詳解
接口是用來(lái)定義行為的類(lèi)型,定義的行為不由接口直接實(shí)現(xiàn),而由通過(guò)方法由定義的類(lèi)型實(shí)現(xiàn),本文就來(lái)和大家詳細(xì)講講Go語(yǔ)言中接口的使用吧2023-03-03Golang中如何使用lua進(jìn)行擴(kuò)展詳解
這篇文章主要給大家介紹了關(guān)于Golang中如何使用lua進(jìn)行擴(kuò)展的相關(guān)資料,這是最近在工作中遇到的一個(gè)問(wèn)題,覺(jué)著有必要分享出來(lái)給大家學(xué)習(xí),文中給出了詳細(xì)的示例,需要的朋友可以參考借鑒,下面來(lái)一起看看吧。2017-10-10Go語(yǔ)言基礎(chǔ)學(xué)習(xí)之map的示例詳解
哈希表是常見(jiàn)的數(shù)據(jù)結(jié)構(gòu),有的語(yǔ)言會(huì)將哈希稱(chēng)作字典或者映射,在Go中,哈希就是常見(jiàn)的數(shù)據(jù)類(lèi)型map,本文就來(lái)聊聊Golang中map的相關(guān)知識(shí)吧2023-04-04輕松構(gòu)建Go應(yīng)用的Dockerfile
本文介紹了如何制作一個(gè)用于構(gòu)建和運(yùn)行Go應(yīng)用程序的Docker鏡像的Dockerfile的相關(guān)資料,需要的朋友可以參考下2023-10-10淺析go中Ticker,Timer和Tick的用法與區(qū)別
在go面試的時(shí)候,面試官經(jīng)常會(huì)問(wèn)time包的Ticker,Timer以及Tick的區(qū)別,一般在超時(shí)控制的時(shí)候用的比較多,今天就跟隨小編一起來(lái)詳細(xì)學(xué)一下這幾個(gè)的區(qū)別吧2023-10-10Go結(jié)構(gòu)體從基礎(chǔ)到應(yīng)用深度探索
本文深入探討了結(jié)構(gòu)體的定義、類(lèi)型、字面量表示和使用方法,旨在為讀者呈現(xiàn)Go結(jié)構(gòu)體的全面視角,通過(guò)結(jié)構(gòu)體,開(kāi)發(fā)者可以實(shí)現(xiàn)更加模塊化、高效的代碼設(shè)計(jì),這篇文章旨在為您提供關(guān)于結(jié)構(gòu)體的深入理解,助您更好地利用Go語(yǔ)言的強(qiáng)大功能2023-10-10