首頁  >  文章  >  後端開發  >  緩衝通道的行為與我對 Go 的期望有何不同?

緩衝通道的行為與我對 Go 的期望有何不同?

PHPz
PHPz轉載
2024-02-09 10:09:21333瀏覽

缓冲通道的行为与我对 Go 的期望有何不同?

在Go語言中,緩衝通道是一種特殊類型的通道,它與普通通道的行為不同。普通通道在傳送資料時,傳送者會被阻塞,直到有接收者接收資料為止。而緩衝通道則允許發送者在通道未滿時繼續發送數據,而不會被阻塞。這樣一來,發送者可以更快地完成發送操作,而不需要等待接收者。對於Go語言的使用者來說,緩衝通道的行為可以提供更高的並發效能和更好的反應速度。

問題內容

我試圖了解緩衝通道的工作原理並為其編寫程式碼片段

package main

import (
    "fmt"
)

func squares(c chan int) {
    for i := 0; i < 4; i++ {
        num := <-c
        fmt.println(num * num)
    }
}

func main() {
    fmt.println("main() started")
    c := make(chan int, 3)
    
        go squares(c)

    c <- 1
    c <- 2
    c <- 3
    c <- 4 // blocks here

    fmt.println("main() stopped")
}

按照我預期程序的行為方式,主 goroutine 啟動並持續到 c<-4,此時它被阻塞,控制權轉到 square goroutine(因為緩衝區容量為 3)。 squares goroutine 中的循環一直持續到第四次迭代,此時通道為空。空通道上的讀取操作會被阻塞,因此控制權會回到主 goroutine。此時,對通道(c<-4)的寫入操作被執行,我們列印“main()已停止”,程式結束。

意思是我期望的輸出是,

main() started
1
4
9
main() stopped

但是我得到了輸出,

main() started
1
4
9
16
main() stopped
如何?我是否遺漏了一些管道到底如何運作的資訊?

解決方法

頻道不是這樣運作的。

Goroutines 並發運行。這意味著當一個 Goroutine 發送到緩衝通道時,另一個等待從該通道接收的 Goroutine 可以立即接收它。它不會等待通道填滿。

至於程式的結束,當你將最後一個數字發送到通道時,不能保證 goroutine 會在程式結束之前拾取它並列印輸出,因為你沒有等待 goroutine完全的。因此,幸運的是,它運行並列印輸出。還會有其他執行不會發生這種情況,程式會在 goroutine 列印輸出之前終止。

以上是緩衝通道的行為與我對 Go 的期望有何不同?的詳細內容。更多資訊請關注PHP中文網其他相關文章!

陳述:
本文轉載於:stackoverflow.com。如有侵權,請聯絡admin@php.cn刪除