Go并发编程中通道数据接收顺序并非Goroutine启动顺序的解析
Go语言中的通道(channel)是Goroutine间高效通信的关键机制。本文将分析一段Go代码,解释为何从通道接收数据的顺序与Goroutine启动顺序不一致。
这段代码展示了两个Goroutine分别计算切片不同部分的和,并将结果发送到同一个通道,主Goroutine最后接收结果。代码如下:
package main import "fmt" func sum(s []int, c chan int) { sum := 0 for _, v := range s { sum += v } c <- sum // 将计算结果发送到通道c } func main() { a := []int{7, 2, 8, -9, 4, 0} c := make(chan int) go sum(a[:len(a)/2], c) // 启动Goroutine 1,计算前一半切片的和 go sum(a[len(a)/2:], c) // 启动Goroutine 2,计算后一半切片的和 x, y := <-c, <-c // 从通道c接收两个结果 fmt.Println(x, y, x+y) // 打印结果 }
登录后复制
代码中,x, y := 同步机制,例如WtGroup或锁。
为了更清晰地说明,可以将代码修改为打印Goroutine的ID:
package main import ( "fmt" "runtime" ) func sum(s []int, c chan int) { sum := 0 for _, v := range s { sum += v } c <- sum } func main() { a := []int{7, 2, 8, -9, 4, 0} c := make(chan int) go func() { fmt.Println("Goroutine 1 ID:", runtime.GoID()) sum(a[:len(a)/2], c) }() go func() { fmt.Println("Goroutine 2 ID:", runtime.GoID()) sum(a[len(a)/2:], c) }() x, y := <-c, <-c fmt.Println(x, y, x+y) }
登录后复制
运行此修改后的代码,可以观察到Goroutine ID与接收到的结果顺序并不一致,从而更直观地理解通道接收的非确定性。
以上就是Go并发编程中,通道接收数据的顺序不是goroutine启动的顺序?的详细内容,更多请关注php中文网其它相关文章!