Go基礎(chǔ)教程系列之WaitGroup用法實例詳解
正常情況下,新激活的goroutine(協(xié)程)的結(jié)束過程是不可控制的,唯一可以保證終止goroutine(協(xié)程)的行為是main goroutine(協(xié)程)的終止。也就是說,我們并不知道哪個goroutine(協(xié)程)什么時候結(jié)束。
但很多情況下,我們正需要知道goroutine(協(xié)程)是否完成。這需要借助sync包的WaitGroup來實現(xiàn)。
WatiGroup是sync包中的一個struct類型,用來收集需要等待執(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個方法:
- Add():每次激活想要被等待完成的goroutine(協(xié)程)之前,先調(diào)用Add(),用來設(shè)置或添加要等待完成的goroutine(協(xié)程)數(shù)量
- 例如Add(2)或者兩次調(diào)用Add(1)都會設(shè)置等待計數(shù)器的值為2,表示要等待2個goroutine(協(xié)程)完成
- Done():每次需要等待的goroutine(協(xié)程)在真正完成之前,應(yīng)該調(diào)用該方法來人為表示goroutine(協(xié)程)完成了,該方法會對等待計數(shù)器減1
- Wait():在等待計數(shù)器減為0之前,Wait()會一直阻塞當(dāng)前的goroutine(協(xié)程)
也就是說,Add()用來增加要等待的goroutine(協(xié)程)的數(shù)量,Done()用來表示goroutine(協(xié)程)已經(jīng)完成了,減少一次計數(shù)器,Wait()用來等待所有需要等待的goroutine(協(xié)程)完成。
下面是一個示例,通過示例很容易理解。
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個goroutine,每次激活goroutine之前,都先調(diào)用Add()方法增加一個需要等待的goroutine計數(shù)。每個goroutine都運行process()函數(shù),這個函數(shù)在執(zhí)行完成時需要調(diào)用Done()方法來表示goroutine的結(jié)束。激活3個goroutine后,main goroutine會執(zhí)行到Wait(),由于每個激活的goroutine運行的process()都需要睡眠2秒,所以main goroutine在Wait()這里會阻塞一段時間(大約2秒),當(dāng)所有g(shù)oroutine都完成后,等待計數(shù)器減為0,Wait()將不再阻塞,于是main goroutine得以執(zhí)行后面的Println()。
還有一點需要特別注意的是process()中使用指針類型的*sync.WaitGroup作為參數(shù),這里不能使用值類型的sync.WaitGroup作為參數(shù),因為這意味著每個goroutine都拷貝一份wg,每個goroutine都使用自己的wg。這顯然是不合理的,這3個goroutine應(yīng)該共享一個wg,才能知道這3個goroutine都完成了。實際上,如果使用值類型的參數(shù),main goroutine將會永久阻塞而導(dǎo)致產(chǎn)生死鎖。
更多關(guān)于Go WaitGroup用法教程請查看下面的相關(guān)鏈接
相關(guān)文章
在 Golang 中實現(xiàn)一個簡單的Http中間件過程詳解
本文在go web中簡單的實現(xiàn)了中間件的機制,這樣帶來的好處也是顯而易見的,當(dāng)然社區(qū)也有一些成熟的 middleware 組件,包括 Gin 一些Web框架中也包含了 middleware 相關(guān)的功能,具體內(nèi)容詳情跟隨小編一起看看吧2021-07-07
Go語言利用aicli實現(xiàn)輕松調(diào)用DeepSeek和ChatGPT
這篇文章主要為大家介紹了一款用Go語言編寫的AI助手客戶端庫——aicli,該庫不僅支持ChatGPT,還集成了DeepSeek,感興趣的小伙伴可以了解一下2025-03-03
GoAdminGroup/go-admin的安裝和運行的教程詳解
這篇文章主要介紹了GoAdminGroup/go-admin的安裝和運行的教程詳解,本文通過實例代碼給大家介紹的非常詳細,對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友可以參考下2020-09-09
GO中的時間操作總結(jié)(time&dateparse)
日常開發(fā)過程中,對于時間的操作可謂是無處不在,但是想實現(xiàn)時間自由還是不簡單的,多種時間格式容易混淆,本文為大家整理了一下GO中的時間操作,有需要的可以參考下2023-09-09
Golang標(biāo)準庫之errors包應(yīng)用方式
Go語言的errors包提供了基礎(chǔ)的錯誤處理能力,允許通過errors.New創(chuàng)建自定義error對象,error在Go中是一個接口,通過實現(xiàn)Error方法來定義錯誤文本,對錯誤的比較通常基于對象地址,而非文本內(nèi)容,因此即使兩個錯誤文本相同2024-10-10

