我正在学习Go,并且遇到了以下代码片段:
package main import "fmt" func sum(a []int, c chan int) { sum := 0 for _, v := range a { sum += v } c <- sum // send sum to c } func main() { a := []int{7, 2, 8, -9, 4, 0} c := make(chan int, 2) go sum(a[0:3], c) go sum(a[3:6], c) x := <-c y := <-c // x, y := <-c, <-c // receive from c fmt.Println(x, y) } Output: -5 17 Program exited.
有人可以告诉我为什么“ sum”函数的第二个调用在第一个调用之前通过通道吗?在我看来,输出应为:
17 -5
我还使用无缓冲通道对此进行了测试,它也给出了相同顺序的输出。我想念什么?
您正在代码中调用go例程,但无法确定例程何时结束并将该值传递到缓冲通道。
由于该代码是异步的,因此只要例程完成,它将把数据写入通道,并在另一侧读取。在上面的示例中,您仅调用两个go例程,因此行为是确定的,并且在大多数情况下都会以某种方式生成相同的输出,但是当您增加go例程时,输出将不会相同,并且顺序会有所不同,除非您执行以下操作它同步。
例:
package main import "fmt" func sum(a []int, c chan int) { sum := 0 for _, v := range a { sum += v } c <- sum // send sum to c } func main() { a := []int{7, 2, 8, -9, 4, 2, 4, 2, 8, 2, 7, 2, 99, -32, 2, 12, 32, 44, 11, 63} c := make(chan int) for i := 0; i < len(a); i = i + 5 { go sum(a[i:i+5], c) } output := make([]int, 5) for i := 0; i < 4; i++ { output[i] = <-c } close(c) fmt.Println(output) }
此代码在不同样本运行中的输出为 [12 18 0 78 162] [162 78 12 0 18] [12 18 78 162 0]
此代码在不同样本运行中的输出为
[12 18 0 78 162] [162 78 12 0 18] [12 18 78 162 0]
这是因为goroutines将输出异步写入缓冲的通道。
希望这可以帮助。