在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中文網其他相關文章!