亚洲乱码中文字幕综合,中国熟女仑乱hd,亚洲精品乱拍国产一区二区三区,一本大道卡一卡二卡三乱码全集资源,又粗又黄又硬又爽的免费视频

Go語言無緩沖的通道的使用

 更新時間:2024年01月05日 09:46:06   作者:C語言中文網(wǎng)  
Go語言中無緩沖的通道是指在接收前沒有能力保存任何值的通道,本文主要介紹了Go語言無緩沖的通道的使用,具有一定的參考價值,感興趣的可以了解一下

Go語言中無緩沖的通道(unbuffered channel)是指在接收前沒有能力保存任何值的通道。這種類型的通道要求發(fā)送 goroutine 和接收 goroutine 同時準備好,才能完成發(fā)送和接收操作。

如果兩個 goroutine 沒有同時準備好,通道會導(dǎo)致先執(zhí)行發(fā)送或接收操作的 goroutine 阻塞等待。這種對通道進行發(fā)送和接收的交互行為本身就是同步的。其中任意一個操作都無法離開另一個操作單獨存在。

阻塞指的是由于某種原因數(shù)據(jù)沒有到達,當(dāng)前協(xié)程(線程)持續(xù)處于等待狀態(tài),直到條件滿足才解除阻塞。

同步指的是在兩個或多個協(xié)程(線程)之間,保持數(shù)據(jù)內(nèi)容一致性的機制。

下圖展示兩個 goroutine 如何利用無緩沖的通道來共享一個值。

使用無緩沖的通道在 goroutine 之間同步

圖:使用無緩沖的通道在 goroutine 之間同步

在第 1 步,兩個 goroutine 都到達通道,但哪個都沒有開始執(zhí)行發(fā)送或者接收。在第 2 步,左側(cè)的 goroutine 將它的手伸進了通道,這模擬了向通道發(fā)送數(shù)據(jù)的行為。這時,這個 goroutine 會在通道中被鎖住,直到交換完成。

在第 3 步,右側(cè)的 goroutine 將它的手放入通道,這模擬了從通道里接收數(shù)據(jù)。這個 goroutine 一樣也會在通道中被鎖住,直到交換完成。在第 4 步和第 5 步,進行交換,并最終在第 6 步,兩個 goroutine 都將它們的手從通道里拿出來,這模擬了被鎖住的 goroutine 得到釋放。兩個 goroutine 現(xiàn)在都可以去做別的事情了。

為了講得更清楚,讓我們來看兩個完整的例子。這兩個例子都會使用無緩沖的通道在兩個 goroutine 之間同步交換數(shù)據(jù)。

【示例 1】在網(wǎng)球比賽中,兩位選手會把球在兩個人之間來回傳遞。選手總是處在以下兩種狀態(tài)之一,要么在等待接球,要么將球打向?qū)Ψ???梢允褂脙蓚€ goroutine 來模擬網(wǎng)球比賽,并使用無緩沖的通道來模擬球的來回,代碼如下所示。

// 這個示例程序展示如何用無緩沖的通道來模擬
// 2 個goroutine 間的網(wǎng)球比賽
package main
import (
    "fmt"
    "math/rand"
    "sync"
    "time"
)
// wg 用來等待程序結(jié)束
var wg sync.WaitGroup
func init() {
    rand.Seed(time.Now().UnixNano())
}
// main 是所有Go 程序的入口
func main() {
    // 創(chuàng)建一個無緩沖的通道
    court := make(chan int)
    // 計數(shù)加 2,表示要等待兩個goroutine
    wg.Add(2)
    // 啟動兩個選手
    go player("Nadal", court)
    go player("Djokovic", court)
    // 發(fā)球
    court <- 1
    // 等待游戲結(jié)束
    wg.Wait()
}
// player 模擬一個選手在打網(wǎng)球
func player(name string, court chan int) {
    // 在函數(shù)退出時調(diào)用Done 來通知main 函數(shù)工作已經(jīng)完成
    defer wg.Done()
    for {
        // 等待球被擊打過來
        ball, ok := <-court
        if !ok {
            // 如果通道被關(guān)閉,我們就贏了
            fmt.Printf("Player %s Won\n", name)
            return
        }
        // 選隨機數(shù),然后用這個數(shù)來判斷我們是否丟球
        n := rand.Intn(100)
        if n%13 == 0 {
            fmt.Printf("Player %s Missed\n", name)
            // 關(guān)閉通道,表示我們輸了
            close(court)
            return
        }
        // 顯示擊球數(shù),并將擊球數(shù)加1
        fmt.Printf("Player %s Hit %d\n", name, ball)
        ball++
        // 將球打向?qū)κ?
        court <- ball
    }
}

運行這個程序,輸出結(jié)果如下所示。

Player Nadal Hit 1
Player Djokovic Hit 2
Player Nadal Hit 3
Player Djokovic Missed
Player Nadal Won

代碼說明如下:

  • 第 22 行,創(chuàng)建了一個 int 類型的無緩沖的通道,讓兩個 goroutine 在擊球時能夠互相同步。
  • 第 28 行和第 29 行,創(chuàng)建了參與比賽的兩個 goroutine。在這個時候,兩個 goroutine 都阻塞住等待擊球。
  • 第 32 行,將球發(fā)到通道里,程序開始執(zhí)行這個比賽,直到某個 goroutine 輸?shù)舯荣悺?/li>
  • 第 43 行可以找到一個無限循環(huán)的 for 語句。在這個循環(huán)里,是玩游戲的過程。
  • 第 45 行,goroutine 從通道接收數(shù)據(jù),用來表示等待接球。這個接收動作會鎖住 goroutine,直到有數(shù)據(jù)發(fā)送到通道里。通道的接收動作返回時。
  • 第 46 行會檢測 ok 標(biāo)志是否為 false。如果這個值是 false,表示通道已經(jīng)被關(guān)閉,游戲結(jié)束。
  • 第 53 行到第 60 行,會產(chǎn)生一個隨機數(shù),用來決定 goroutine 是否擊中了球。
  • 第 58 行如果某個 goroutine 沒有打中球,關(guān)閉通道。之后兩個 goroutine 都會返回,通過 defer 聲明的 Done 會被執(zhí)行,程序終止。
  • 第 64 行,如果擊中了球 ball 的值會遞增 1,并在第 67 行,將 ball 作為球重新放入通道,發(fā)送給另一位選手。在這個時刻,兩個 goroutine 都會被鎖住,直到交換完成。

【示例 2】用不同的模式,使用無緩沖的通道,在 goroutine 之間同步數(shù)據(jù),來模擬接力比賽。在接力比賽里,4 個跑步者圍繞賽道輪流跑。第二個、第三個和第四個跑步者要接到前一位跑步者的接力棒后才能起跑。比賽中最重要的部分是要傳遞接力棒,要求同步傳遞。在同步接力棒的時候,參與接力的兩個跑步者必須在同一時刻準備好交接。代碼如下所示。

// 這個示例程序展示如何用無緩沖的通道來模擬
// 4 個goroutine 間的接力比賽
package main
import (
    "fmt"
    "sync"
    "time"
)
// wg 用來等待程序結(jié)束
var wg sync.WaitGroup
// main 是所有Go 程序的入口
func main() {
    // 創(chuàng)建一個無緩沖的通道
    baton := make(chan int)
    // 為最后一位跑步者將計數(shù)加1
    wg.Add(1)
    // 第一位跑步者持有接力棒
    go Runner(baton)
    // 開始比賽
    baton <- 1
    // 等待比賽結(jié)束
    wg.Wait()
}
// Runner 模擬接力比賽中的一位跑步者
func Runner(baton chan int) {
    var newRunner int
    // 等待接力棒
    runner := <-baton
    // 開始繞著跑道跑步
    fmt.Printf("Runner %d Running With Baton\n", runner)
    // 創(chuàng)建下一位跑步者
    if runner != 4 {
        newRunner = runner + 1
        fmt.Printf("Runner %d To The Line\n", newRunner)
        go Runner(baton)
    }
    // 圍繞跑道跑
    time.Sleep(100 * time.Millisecond)
    // 比賽結(jié)束了嗎?
    if runner == 4 {
        fmt.Printf("Runner %d Finished, Race Over\n", runner)
        wg.Done()
        return
    }
    // 將接力棒交給下一位跑步者
    fmt.Printf("Runner %d Exchange With Runner %d\n",
        runner,
        newRunner)
    baton <- newRunner
}

運行這個程序,輸出結(jié)果如下所示。

Runner 1 Running With Baton
Runner 1 To The Line
Runner 1 Exchange With Runner 2
Runner 2 Running With Baton
Runner 2 To The Line
Runner 2 Exchange With Runner 3
Runner 3 Running With Baton
Runner 3 To The Line
Runner 3 Exchange With Runner 4
Runner 4 Running With Baton
Runner 4 Finished, Race Over

代碼說明如下:

  • 第 17 行,創(chuàng)建了一個無緩沖的 int 類型的通道 baton,用來同步傳遞接力棒。
  • 第 20 行,我們給 WaitGroup 加 1,這樣 main 函數(shù)就會等最后一位跑步者跑步結(jié)束。
  • 第 23 行創(chuàng)建了一個 goroutine,用來表示第一位跑步者來到跑道。
  • 第 26 行,將接力棒交給這個跑步者,比賽開始。
  • 第 29 行,main 函數(shù)阻塞在 WaitGroup,等候最后一位跑步者完成比賽。
  • 第 37 行,goroutine 對 baton 通道執(zhí)行接收操作,表示等候接力棒。
  • 第 46 行,一旦接力棒傳了進來,就會創(chuàng)建一位新跑步者,準備接力下一棒,直到 goroutine 是第四個跑步者。
  • 第 50 行,跑步者圍繞跑道跑 100 ms。
  • 第 55 行,如果第四個跑步者完成了比賽,就調(diào)用 Done,將 WaitGroup 減 1,之后 goroutine 返回。
  • 第 64 行,如果這個 goroutine 不是第四個跑步者,接力棒會交到下一個已經(jīng)在等待的跑步者手上。在這個時候,goroutine 會被鎖住,直到交接完成。

在這兩個例子里,我們使用無緩沖的通道同步 goroutine,模擬了網(wǎng)球和接力賽。代碼的流程與這兩個活動在真實世界中的流程完全一樣,這樣的代碼很容易讀懂。

現(xiàn)在知道了無緩沖的通道是如何工作的,下一節(jié)我們將為大家介紹帶緩沖的通道。

到此這篇關(guān)于Go語言無緩沖的通道的使用的文章就介紹到這了,更多相關(guān)Go 無緩沖通道內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • 在Golang代碼中如何自動生成版本號的方法示例

    在Golang代碼中如何自動生成版本號的方法示例

    這篇文章主要給大家介紹了在Golang代碼中如何自動生成版本號的相關(guān)資料,文中通過示例代碼介紹的非常詳細,對大家具有一定的參考學(xué)習(xí)價值,需要的朋友們下面來一起看看吧。
    2017-07-07
  • 使用Go語言進行安卓開發(fā)的詳細教程

    使用Go語言進行安卓開發(fā)的詳細教程

    本文將介紹如何使用Go語言進行安卓開發(fā),我們將探討使用Go語言進行安卓開發(fā)的優(yōu)點、準備工作、基本概念和示例代碼,通過本文的學(xué)習(xí),你將了解如何使用Go語言構(gòu)建高效的安卓應(yīng)用程序,需要的朋友可以參考下
    2023-11-11
  • Go語言中比較兩個map[string]interface{}是否相等

    Go語言中比較兩個map[string]interface{}是否相等

    本文主要介紹了Go語言中比較兩個map[string]interface{}是否相等,我們可以將其轉(zhuǎn)化成順序一樣的 slice ,然后再轉(zhuǎn)化未json,具有一定的參考價值,感興趣的可以了解一下
    2023-08-08
  • Golang對sqlite3數(shù)據(jù)庫進行操作實踐記錄

    Golang對sqlite3數(shù)據(jù)庫進行操作實踐記錄

    sqlite是嵌入式關(guān)系型數(shù)據(jù)庫引擎,官方描述為自包含的、無服務(wù)的、零配置并支持事務(wù)的關(guān)系型數(shù)據(jù)庫引擎,下面這篇文章主要給大家介紹了關(guān)于Golang對sqlite3數(shù)據(jù)庫進行操作的相關(guān)資料,需要的朋友可以參考下
    2024-03-03
  • Go標(biāo)準庫常見錯誤分析和解決辦法

    Go標(biāo)準庫常見錯誤分析和解決辦法

    Go語言的標(biāo)準庫為開發(fā)者提供了豐富且高效的工具,涵蓋了從網(wǎng)絡(luò)編程到文件操作等各個方面,然而,標(biāo)準庫雖好,使用不當(dāng)卻可能適得其反,正所謂"工欲善其事,必先利其器",本文將深入剖析Go標(biāo)準庫使用中的常見錯誤,幫助開發(fā)者避開這些坑,寫出更加健壯的代碼
    2025-04-04
  • Golang實現(xiàn)將視頻按照時間維度剪切的工具

    Golang實現(xiàn)將視頻按照時間維度剪切的工具

    這篇文章主要為大家詳細介紹了如何利用Golang實現(xiàn)將視頻按照時間維度進行剪切,文中的示例代碼講解詳細,感興趣的小伙伴可以跟隨小編一起了解一下
    2022-12-12
  • Go語言Grpc?Stream的實現(xiàn)

    Go語言Grpc?Stream的實現(xiàn)

    本文主要介紹了Go語言Grpc?Stream的實現(xiàn),文中通過示例代碼介紹的非常詳細,對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2022-06-06
  • Golang字符串拼接的性能以及原理詳解

    Golang字符串拼接的性能以及原理詳解

    最近在做性能優(yōu)化,有個函數(shù)里面的耗時特別長,看里面的操作大多是一些字符串拼接的操作,而字符串拼接在golang里面其實有很多種實現(xiàn),下面這篇文章主要給大家介紹了關(guān)于Golang字符串拼接的性能以及原理的相關(guān)資料,需要的朋友可以參考下
    2023-06-06
  • Go語言實現(xiàn)釘釘發(fā)送通知

    Go語言實現(xiàn)釘釘發(fā)送通知

    本文通過代碼給大家介紹了Go語言實現(xiàn)釘釘發(fā)送通知,代碼簡單易懂,非常不錯,具有一定的參考借鑒價值,需要的朋友可以參考下
    2019-11-11
  • Go語言并發(fā)范式之future模式詳解

    Go語言并發(fā)范式之future模式詳解

    編程中經(jīng)常遇到在一個流程中需要調(diào)用多個子調(diào)用的情況,此時就可以使用Go并發(fā)編程中的future模式,下面小編就來和大家聊聊future模式的具體使用,需要的可以參考一下
    2023-06-06

最新評論