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

GO中sync包自由控制并發(fā)示例詳解

 更新時間:2022年08月04日 16:01:45   作者:六號積極分子  
這篇文章主要為大家介紹了GO中sync包自由控制并發(fā)示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪

資源競爭

channel 常用于并發(fā)通信,要保證并發(fā)安全,主要使用互斥鎖。在并發(fā)的過程中,當(dāng)一個內(nèi)存被多個 goroutine 同時訪問時,就會產(chǎn)生資源競爭的情況。這塊內(nèi)存也可以稱為共享資源。

并發(fā)時對于共享資源必然會出現(xiàn)搶占資源的情況,如果是對某資源的統(tǒng)計,很可能就會導(dǎo)致結(jié)果錯誤。為保證只有一個協(xié)程拿到資源并操作它,可以引入互斥鎖 sync.Mutex。

sync.Mutex

互斥鎖,指的是并發(fā)時,在同一時刻只有一個協(xié)程執(zhí)行某段代碼,其他協(xié)程只有等待該協(xié)程執(zhí)行完成后才能繼續(xù)執(zhí)行。

var (sum int 
 mutex sync.Mutex)
func add(i int){
    mutex.Lock()
    sum+=i
    mute.Unlock()
}

使用 mutex 加鎖保護 sum+ =i 的代碼片段,這樣這個片段區(qū)就無法同時被多個協(xié)程訪問,當(dāng)有協(xié)程進入該片段區(qū),那其他的協(xié)程就只有等待,以此保證臨界區(qū)的并發(fā)安全。

sync.Mutex 只有 Lock()和 Unlock() 方法,它們是成對存在的,且Lock后一定要執(zhí)行Unlock進行釋放鎖。所以可以使用 defer 語句釋放鎖,以保證鎖一定會被釋放。

func add(i int){
    mutex.Lock()
    defer mutex.Unlock()
    sum += i
}

sync.RWMutex

上面例子是對 sum 寫操作時使用sync.Mutex 保證并發(fā)安全,當(dāng)多個協(xié)程進行讀取操作時,避免因并發(fā)產(chǎn)生讀取數(shù)據(jù)不正確,也是可以使用互斥鎖 sync.Mutex。

func getSum(){
    mutex.Lock()
    defer mutex.Unlock()
    b:=sum
    return b
}

多個協(xié)程 goroutine 同時讀寫的資源競爭問題解決,還需要考慮性能問題,每次讀寫共享資源都加鎖,也會導(dǎo)致性能低。

多個協(xié)程并發(fā)進行讀寫時會遇到以下情況:

  • 寫時不能同時讀,易讀到臟數(shù)據(jù)
  • 讀時不能同時寫,因為會導(dǎo)致結(jié)果不一致
  • 讀時同時寫,因數(shù)據(jù)不變,無論多少個 goroutine 讀都是并發(fā)安全

使用讀寫鎖 sync.RWMutex 優(yōu)化代碼:

var mutex sync.RWMutex
func readSum() int {
    mutex.RLock()
    defer mutex.RUnlock()
    b := sum
    return b
}

讀寫鎖的性能比互斥鎖性能好。

sync.WaitGroup

為了能夠監(jiān)聽所有的協(xié)程的執(zhí)行,一旦所有的goroutine 都執(zhí)行完成,程序應(yīng)當(dāng)及時退出節(jié)省時間提高性能。通過使用 sync.WaitGroup 來解決。使用步驟如下:

  • 聲明一個 sync.WaitGroup ,通過 add 方法增加計數(shù)器值,有幾個協(xié)程就計算幾個
  • 每個協(xié)程結(jié)束后就調(diào)用 Done 方法,主要是讓計數(shù)器減1
  • 最后調(diào)用 Wait 方法一直等待,直到計數(shù)器為 0 時,所有跟蹤的協(xié)程都執(zhí)行完畢
func run() {
    var wg sync.WaitGroup
    wg.Add(100)
    for i := 0; i < 100; i++ {
        go func() {
            defer wg.Done()
            add(10)
        }()
    }
    wg.Wait()
}

通過 sync.WaitGroup 可以很好地跟蹤協(xié)程.

sync.Once

sync.Once 作用是讓代碼只執(zhí)行一次。詳細使用是調(diào)用方法 once.Do 方法,具體實現(xiàn):

func main(){
    var once sync.once
    oneFunc := func(){
        println("once func")
    }
    once.Do(oneFunc)
}

sync.Once 適用于創(chuàng)建某個對象的單例、只加載一次的資源等只執(zhí)行一次的場景。

sync.Cond

使用 sync.WaitGroup 主要是控制等待所有的協(xié)程都執(zhí)行完畢,才最終完成。但是當(dāng)遇到場景是,只有等待所有條件都準備好才開始。sync.Cond 相當(dāng)于發(fā)號施令,只有通知執(zhí)行所有的協(xié)程才可以執(zhí)行,重點是所有協(xié)程需等待喚醒才可以開始。

所以 sync.Cond 具有阻塞協(xié)程和喚醒協(xié)程的功能。詳細的用法:

  • 通過 sync.NewCond 函數(shù)生成一個 *sync.Cond,用于阻塞和喚醒協(xié)程
  • 調(diào)用 cond.Wait() 方法阻塞當(dāng)前協(xié)程等待,需要注意調(diào)用 cond.Wait() 方法要加鎖
  • 調(diào)用 cond.Broadcast() 后所有協(xié)程才開始執(zhí)行
func run() {
    cond := sync.NewCond(&amp;sync.Mutex{})
    var wg sync.WaitGroup
    wg.Add(101)
    for i := 0; i &lt; 100; i++ {
        go func(num int) {
            defer wg.Done()
            fmt.Println(num, "號正在 awaiting......")
            cond.L.Lock()
            cond.Wait() //等待所有協(xié)程準備完成
            fmt.Println(num, "號開始跑……")
            cond.L.Unlock()
        }(i)
    }
    // 等待所有的協(xié)程都進入 wait 狀態(tài)
    time.Sleep(2*time.Second)
    go func() {
        defer wg.Done()
        // 所有都準備完成,開始
        cond.Broadcast()
    }()
    // 防止函數(shù)提前返回退出
    wg.Wait()
}

以上就是GO中sync包自由控制并發(fā)示例詳解的詳細內(nèi)容,更多關(guān)于GO sync包控制并發(fā)的資料請關(guān)注腳本之家其它相關(guān)文章!

相關(guān)文章

  • golang 打印error的堆棧信息操作

    golang 打印error的堆棧信息操作

    這篇文章主要介紹了golang 打印error的堆棧信息操作,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧
    2021-05-05
  • Golang報“import cycle not allowed”錯誤的2種解決方法

    Golang報“import cycle not allowed”錯誤的2種解決方法

    這篇文章主要給大家介紹了關(guān)于Golang報"import cycle not allowed"錯誤的2種解決方法,文中通過示例代碼介紹的非常詳細,對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友可以們下面隨著小編來一起看看吧
    2018-08-08
  • Go語言TCP從原理到代碼實現(xiàn)詳解

    Go語言TCP從原理到代碼實現(xiàn)詳解

    這篇文章主要為大家介紹了Go語言TCP從原理到代碼實現(xiàn)詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪
    2022-08-08
  • 通過案例簡單聊聊為什么說Go中的字符串是不能被修改的

    通過案例簡單聊聊為什么說Go中的字符串是不能被修改的

    在接觸Go這么語言,可能你經(jīng)常會聽到這樣一句話,對于字符串不能修改,可能你很納悶,日常開發(fā)中我們對字符串進行修改也是很正常的,為什么又說Go中的字符串不能進行修改呢,本文就來通過實際案例給大家演示,為什么Go中的字符串不能進行修改
    2023-07-07
  • go語言中切片的長度和容量的區(qū)別

    go語言中切片的長度和容量的區(qū)別

    這篇文章主要介紹了go語言中切片的長度和容量的區(qū)別,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧
    2021-04-04
  • go1.8之安裝配置具體步驟

    go1.8之安裝配置具體步驟

    下面小編就為大家?guī)硪黄猤o1.8之安裝配置具體步驟。小編覺得挺不錯的,現(xiàn)在就分享給大家,也給大家做個參考。一起跟隨小編過來看看吧
    2017-06-06
  • Go高效率開發(fā)Web參數(shù)校驗三種方式實例

    Go高效率開發(fā)Web參數(shù)校驗三種方式實例

    這篇文章主要介紹了Go高效率開發(fā)Web參數(shù)校驗三種方式實例,需要的朋友可以參考下
    2022-11-11
  • Go語言中的各類運算操作符詳解

    Go語言中的各類運算操作符詳解

    本文全面探討了Go語言中的各類運算操作符,從基礎(chǔ)的數(shù)學(xué)和位運算到邏輯和特殊運算符,文章旨在深入解析每一種運算操作符的工作原理、應(yīng)用場景和注意事項,以幫助開發(fā)者編寫更高效、健壯和可讀的Go代碼,</P><P>
    2023-09-09
  • 深入理解Go Gin框架中間件的實現(xiàn)原理

    深入理解Go Gin框架中間件的實現(xiàn)原理

    在Go Gin框架中,中間件是一種在請求處理過程中插入的功能模塊,它可以用于處理請求的前置和后置邏輯,例如認證、日志記錄、錯誤處理等,本文將給大家介紹一下Go Gin框架中間件的實現(xiàn)原理,需要的朋友可以參考下
    2023-09-09
  • Go類型安全的HTTP請求示例詳解

    Go類型安全的HTTP請求示例詳解

    這篇文章主要為大家介紹了Go類型安全的HTTP請求示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪
    2022-06-06

最新評論