go 语言并发编程:通道接收顺序的非确定性
本文分析 Go 语言并发编程中,使用通道 (channel) 接收数据的顺序问题。以下代码片段阐述了一个常见的场景,并解释了接收顺序的不确定性:
<code class="go">package main
import "fmt"
func sum(s []int, c chan int) {
sum := 0
for _, v := range s {
sum += v
}
c <- sum
}
func main() {
s := []int{7, 2, 8, -9, 4, 0}
c := make(chan int)
go sum(s[:len(s)/2], c)
go sum(s[len(s)/2:], c)
x, y := <-c, <-c // 关键点:赋值运算符的结合性
fmt.Println(x, y)
}</code>上述代码中,两个 goroutine 并发计算切片 s 的不同部分的和,并将结果发送到通道 c。main 函数从通道 c 接收两个值,分别赋值给变量 x 和 y。然而,输出结果 x 和 y 的值并非总是与 goroutine 启动顺序一致。例如,可能出现 x 为第二个 goroutine 的结果,y 为第一个 goroutine 的结果的情况。
这并非因为通道本身的接收顺序不确定,而是由于 Go 语言中 赋值运算符的右结合性。表达式 x, y := <-c, <-c 等价于 x, y := (<-, c), (<-, c)。这意味着,<-c 先被执行两次,然后将结果分别赋值给 x 和 y。由于 goroutine 的执行顺序是并发且非确定的,因此从通道 c 接收数据的顺序也无法预知。

为了确保接收顺序,需要使用其他同步机制,例如使用 select 语句或其他协调方法来控制接收顺序。
总而言之,Go 语言通道本身并不保证接收顺序,赋值运算符的右结合性以及 goroutine 的并发执行特性共同导致了接收顺序的不确定性。 开发者应该理解并避免依赖于通道的接收顺序,而是使用更可靠的同步机制来处理并发场景。
以上就是Go并发编程中,通道接收顺序为何不确定?的详细内容,更多请关注php中文网其它相关文章!
每个人都需要一台速度更快、更稳定的 PC。随着时间的推移,垃圾文件、旧注册表数据和不必要的后台进程会占用资源并降低性能。幸运的是,许多工具可以让 Windows 保持平稳运行。
Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号