Go sync.WaitGroup源碼閱讀

WaitGroup并發(fā)控制

sync.WaitGroup內(nèi)部維護著一個計數(shù)器,計數(shù)器的值可以增加和減少。

例如當我們啟動了N 個并發(fā)任務時,就通過Add()將計數(shù)器值增加N。

每個任務完成通過調(diào)用Done()方法將計數(shù)器減1。

通過調(diào)用Wait()來等待并發(fā)任務執(zhí)行完,當計數(shù)器值為0時,表示所有并發(fā)任務已經(jīng)完成。

const N = 10

var wg = &sync.WaitGroup{}

func main() {

    for i := 0; i < N; i++ {
        //wg.Add(1) 正確寫法
        go func(i int) {
            wg.Add(1)
            println(i)
            defer wg.Done()
        }(i)
    }
    wg.Wait()

}

結(jié)果

結(jié)果不唯一,代碼存在風險, 所有g(shù)o未必都能執(zhí)行到

這是使用WaitGroup經(jīng)常犯下的錯誤!請各位同學多次運行就會發(fā)現(xiàn)輸出都會不同甚至又出現(xiàn)報錯的問題。 這是因為go執(zhí)行太快了,導致wg.Add(1)還沒有執(zhí)行main函數(shù)就執(zhí)行完畢了。

底層原理

type WaitGroup struct {
   noCopy noCopy

   state1 [3]uint32  // uint32數(shù)組一共12個字節(jié),前8個即uint64記錄 高8位記錄需要等待的數(shù)量 低8位正在等待                       的數(shù)量  ,后4個字節(jié)存儲信號量,用于喚醒
}

1.核心原理就是通過之前說的64位的uint64來進行計數(shù),采用高位記錄需要Done的數(shù)量,低位記錄Wait的數(shù)量,然后排隊休眠等待喚醒

2.如果發(fā)現(xiàn)當前count>0則 Wait的goroutine會進行排隊

3.任務完成后的goroutine則進行Done操作,直到count==0,則完成,就喚醒所有因為wait操作睡眠的goroutine

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
【社區(qū)內(nèi)容提示】社區(qū)部分內(nèi)容疑似由AI輔助生成,瀏覽時請結(jié)合常識與多方信息審慎甄別。
平臺聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點,簡書系信息發(fā)布平臺,僅提供信息存儲服務。

相關(guān)閱讀更多精彩內(nèi)容

友情鏈接更多精彩內(nèi)容