欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

Go并發(fā):使用sync.WaitGroup實(shí)現(xiàn)協(xié)程同步方式

 更新時間:2021年05月04日 09:19:09   作者:Cynhard85  
這篇文章主要介紹了Go并發(fā):使用sync.WaitGroup實(shí)現(xiàn)協(xié)程同步方式,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧

經(jīng)??吹接腥藭柸绾蔚却鲄f(xié)程中創(chuàng)建的協(xié)程執(zhí)行完畢之后再結(jié)束主協(xié)程,例如如下代碼:

package main
import (
    "fmt"
)
func main() {
    go func() {
        fmt.Println("Goroutine 1")
    }()
    go func() {
        fmt.Println("Goroutine 2")
    }()
}

執(zhí)行以上代碼很可能看不到輸出,因?yàn)橛锌赡苓@兩個協(xié)程還沒得到執(zhí)行主協(xié)程已經(jīng)結(jié)束了,而主協(xié)程結(jié)束時會結(jié)束所有其他協(xié)程。

解決辦法是可以在main函數(shù)結(jié)尾加上等待:

package main
import (
    "fmt"
    "time"
)
func main() {
    go func() {
        fmt.Println("Goroutine 1")
    }()
    go func() {
        fmt.Println("Goroutine 2")
    }()
    time.Sleep(time.Second * 1) // 睡眠1秒,等待上面兩個協(xié)程結(jié)束
}

這并不是完美的解決方法,如果這兩個協(xié)程中包含復(fù)雜的操作,可能很耗時間,就無法確定需要睡眠多久,當(dāng)然可以用管道實(shí)現(xiàn)同步:

package main
import (
    "fmt"
)
func main() {
    ch := make(chan struct{})
    count := 2 // count 表示活動的協(xié)程個數(shù)
    go func() {
        fmt.Println("Goroutine 1")
        ch <- struct{}{} // 協(xié)程結(jié)束,發(fā)出信號
    }()
    go func() {
        fmt.Println("Goroutine 2")
        ch <- struct{}{} // 協(xié)程結(jié)束,發(fā)出信號
    }()
    for range ch {
        // 每次從ch中接收數(shù)據(jù),表明一個活動的協(xié)程結(jié)束
        count--
        // 當(dāng)所有活動的協(xié)程都結(jié)束時,關(guān)閉管道
        if count == 0 {
            close(ch)
        }
    }
}

上面的解決方案是比較完美的方案,但是Go提供了更簡單的方法——使用sync.WaitGroup。

WaitGroup顧名思義,就是用來等待一組操作完成的。

WaitGroup內(nèi)部實(shí)現(xiàn)了一個計(jì)數(shù)器,用來記錄未完成的操作個數(shù),它提供了三個方法,Add()用來添加計(jì)數(shù)。

Done()用來在操作結(jié)束時調(diào)用,使計(jì)數(shù)減一。

Wait()用來等待所有的操作結(jié)束,即計(jì)數(shù)變?yōu)?,該函數(shù)會在計(jì)數(shù)不為0時等待,在計(jì)數(shù)為0時立即返回。

package main
import (
    "fmt"
    "sync"
)
func main() {
    var wg sync.WaitGroup
    wg.Add(2) // 因?yàn)橛袃蓚€動作,所以增加2個計(jì)數(shù)
    go func() {
        fmt.Println("Goroutine 1")
        wg.Done() // 操作完成,減少一個計(jì)數(shù)
    }()
    go func() {
        fmt.Println("Goroutine 2")
        wg.Done() // 操作完成,減少一個計(jì)數(shù)
    }()
    wg.Wait() // 等待,直到計(jì)數(shù)為0
}

可見用sync.WaitGroup是最簡單的方式。

補(bǔ)充:Golang 中使用WaitGroup的那點(diǎn)坑

sync.WaitGroup對于Golang開發(fā)者來說并不陌生,其經(jīng)常作為多協(xié)程之間同步的一種機(jī)制。用好它勢必會讓你事半功倍,但是一旦錯用將引發(fā)問題。

關(guān)于WaitGroup的使用網(wǎng)上有很多例子,在此就不做介紹了,我想說的是我在項(xiàng)目中使用WaitGroup遇到的坑。

在項(xiàng)目中,因?yàn)榉?wù)器有同步需求, 所以直接使用了WaitGroup,但是未考慮使用場景,結(jié)果在項(xiàng)目上線之后,高峰期的時候客戶端經(jīng)常出現(xiàn)卡頓,經(jīng)過多方查找,才發(fā)現(xiàn)如果使用WaitGroup的時候,未啟動單獨(dú)的goroutine,那么極有可能造成主線程的阻塞,

所以我做了下面的測試(測試中,我把WaitGroup置于協(xié)程內(nèi)):

import (
 "fmt"
 "sync"
 "time"
)
 
func main() {
    fmt.Println("main-1")
 testW()
 fmt.Println("main-2")
 time.Sleep(time.Duration(15) * time.Second) 
}
 
func testW() {
 fmt.Println("testW-1")
 go func() {
  var wg sync.WaitGroup
  fmt.Println("testW-2")
  testW1(&wg)
  fmt.Println("testW-5")
  wg.Wait()
  fmt.Println("testW-6")
 }()
}
 
func testW1(wg *sync.WaitGroup) {
 wg.Add(1)
 fmt.Println("testW-3")
 time.AfterFunc(time.Second*5, func() {
  wg.Done()
 })
 fmt.Println("testW-4") 
}

輸出為:

main-1

testchan-1

main-2

testchan-2

testchan-3

testchan-4

testchan-5

// 過5秒

testchan-6

總結(jié):

將WaitGroup用于goroutine內(nèi),不會導(dǎo)致主線程的阻塞,同樣可以實(shí)現(xiàn)同步的效果。

以上為個人經(jīng)驗(yàn),希望能給大家一個參考,也希望大家多多支持腳本之家。如有錯誤或未考慮完全的地方,望不吝賜教。

相關(guān)文章

  • Golang?template?包基本原理分析

    Golang?template?包基本原理分析

    這篇文章主要為大家介紹了Golang?template?包基本原理分析,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪
    2022-09-09
  • Goland 關(guān)閉自動移除未使用的包操作

    Goland 關(guān)閉自動移除未使用的包操作

    這篇文章主要介紹了Goland 關(guān)閉自動移除未使用的包操作,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧
    2020-12-12
  • Go語言編譯原理之變量捕獲

    Go語言編譯原理之變量捕獲

    這篇文章主要為大家介紹了Go語言編譯原理之變量捕獲示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪
    2022-08-08
  • Go 語言 JSON 標(biāo)準(zhǔn)庫的使用

    Go 語言 JSON 標(biāo)準(zhǔn)庫的使用

    今天通過本文給大家介紹Go 語言 JSON 標(biāo)準(zhǔn)庫的使用小結(jié),包括序列化和反序列化的相關(guān)知識,感興趣的朋友跟隨小編一起看看吧
    2021-10-10
  • Golang的Fork/Join實(shí)現(xiàn)代碼

    Golang的Fork/Join實(shí)現(xiàn)代碼

    Fork/Join本質(zhì)上是一種任務(wù)分解,將一個很大的任務(wù)分解成若干個小任務(wù),然后再對小任務(wù)進(jìn)一步分解,直到最小顆粒度,然后并發(fā)執(zhí)行,對Golang的Fork/Join實(shí)現(xiàn)代碼感興趣的朋友跟隨小編一起看看吧
    2023-01-01
  • 一文帶你掌握掌握 Golang結(jié)構(gòu)體與方法

    一文帶你掌握掌握 Golang結(jié)構(gòu)體與方法

    在 Golang 中,結(jié)構(gòu)體和方法是實(shí)現(xiàn)面向?qū)ο缶幊痰闹匾M成部分,也是 Golang 的核心概念之一。在本篇文章中,我們將深入介紹 Golang 結(jié)構(gòu)體與方法的概念、使用方法以及相關(guān)的編程技巧和最佳實(shí)踐
    2023-04-04
  • Go結(jié)構(gòu)體的基本使用詳解

    Go結(jié)構(gòu)體的基本使用詳解

    本文主要介紹了Go結(jié)構(gòu)體的基本使用詳解,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2023-02-02
  • 細(xì)說Go語言中空結(jié)構(gòu)體的奇妙用途

    細(xì)說Go語言中空結(jié)構(gòu)體的奇妙用途

    Go語言中,我們可以定義空結(jié)構(gòu)體,即沒有任何成員變量的結(jié)構(gòu)體,使用關(guān)鍵字?struct{}?來表示。這種結(jié)構(gòu)體似乎沒有任何用處,但實(shí)際上它在?Go?語言中的應(yīng)用非常廣泛,本文就來詳解講講
    2023-05-05
  • golang中一種不常見的switch語句寫法示例詳解

    golang中一種不常見的switch語句寫法示例詳解

    這篇文章主要介紹了golang中一種不常見的switch語句寫法,本文通過示例代碼給大家介紹的非常詳細(xì),對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友可以參考下
    2023-05-05
  • Go map定義的方式及修改技巧

    Go map定義的方式及修改技巧

    這篇文章主要給大家介紹了關(guān)于Go map定義的方式及修改技巧,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2021-02-02

最新評論