golang 中 channel 的一个问题

2022-02-19 11:14:47 +08:00
 mantis
func main() {
    ch := make(chan int, 1)
    for i := 0; i < 10; i++ {
	select {
		case x := <-ch:
			fmt.Println(x)
		case ch <- i:
			fmt.Println("---", i)
		}
	}
}

为什么输出的是:

--- 0
0
--- 2
2
--- 4
4
--- 6
6
--- 8
8
2130 次点击
所在节点    Go 编程语言
9 条回复
labulaka521
2022-02-19 11:17:52 +08:00
chan 容量为 1 ,只能进一个再出一个
Goat121
2022-02-19 11:34:49 +08:00
select 的 case 块之间,如果有多个 case 满足条件,会随机选择一个 case 块执行。
i = 1 的时候,选择了 case x := <-ch: 执行,下面的 case ch <- i 就不再执行,以此类推。实际开发中,chan 的入队和出队一般是由两个 goroutine 分别执行的
Goat121
2022-02-19 11:37:09 +08:00
@Goat121 纠正一下 这里随机选择的特性没生效,i 为奇数时,只满足 case x := <-ch:
daokedao
2022-02-19 12:09:41 +08:00
为 1 3 5... 的时候,输出上次的 0 2 4...
biubiuF
2022-02-19 12:16:27 +08:00
i = 1 ,3 ,5 的时候 ch 里有对象了,执行的 case x := <-ch:
HeliumTang
2022-02-19 12:49:23 +08:00
偶数的时候往 ch 里写,奇数的时候读出来。所以跳过了
lanlanye
2022-02-19 16:12:50 +08:00
你可以在这里找到详细的解释: [基于 select 的多路复用]( https://book.itsfun.top/gopl-zh/ch8/ch8-07.html)
Hanggi
2022-02-19 18:30:39 +08:00
把每一个步骤打出来就好了:

循环 i = 0:
因为 ch 空,所以 x := <-ch 不会执行。
ch <- i 往通道里输入 0
输出 --- 0

循环 i = 1:
因为 ch 里面有数据,所以 x := <- ch 得到上一次压入的 0
输出 0

循环 i = 2:
因为 ch 空,所以 x := <-ch 不会执行。
ch <- i 往通道里输入 2
输出 --- 2

以此类推。。。
kwanzaa
2022-02-19 18:48:25 +08:00
0: ch <- 0
1: x <- ch
2: ch <- 2
3: x <- ch
...

这是一个专为移动设备优化的页面(即为了让你能够在 Google 搜索结果里秒开这个页面),如果你希望参与 V2EX 社区的讨论,你可以继续到 V2EX 上打开本讨论主题的完整版本。

https://www.v2ex.com/t/834983

V2EX 是创意工作者们的社区,是一个分享自己正在做的有趣事物、交流想法,可以遇见新朋友甚至新机会的地方。

V2EX is a community of developers, designers and creative people.

© 2021 V2EX