我有这个程序:package mainimport ( "fmt" "time")var ch1 = make(chan int)var ch2 = make(chan int)func f1() { select { case <-ch1: fmt.Println("ch1") }}func f2() { select { case <-ch2: fmt.Println("ch2") }}func main() { go f1() go f2() time.Sleep(2 * time.Second) fmt.Println("no buffered channel will wait?") ch1 <- 1 ch2 <- 2 fmt.Println("main exits")}我预计,只要 f1 和 f2 不打印任何内容,就意味着 ch1 和 ch2 内部没有任何内容,因此ch1<-1andch2<-2应该阻塞?但运行时,它会打印:no buffered channel will wait?main exits为什么那些无缓冲的通道在主通道中ch1没有ch2被阻塞?如果我不调用f1/ f2in main,就会报错dead lock。我不明白 f1/f2 对 ch1/ch2 做了什么。您能帮忙解释一下他们的行为吗?
1 回答
ABOUTYOU
TA贡献1812条经验 获得超5个赞
和 都有f1()
接收f2()
操作。这些是阻塞操作:只要通道上没有人发送任何内容,它们就会等待。
所以你启动f1()
并f2()
作为新的 goroutine,然后main()
睡觉。同时f1()
和f2()
正在等待来自ch1
和的数据ch2
。
然后main()
醒来,并尝试在 上发送一个值ch1
。这是可以的,因为有一个 goroutine 准备好从它接收数据 ( f1()
)。然后main()
尝试发送ch2
,这也可以,已经f2()
准备好接收了。
然后main()
返回,应用程序结束(它不等待其他 goroutine 打印)。
如果您不启动f1()
并且f2()
作为新的 goroutine,当main()
到达发送语句时,将没有人准备好从通道接收数据,并且由于它是无缓冲的,因此它将阻塞。由于不会再有任何 goroutine 运行,因此这是一个僵局。
- 1 回答
- 0 关注
- 108 浏览
添加回答
举报
0/150
提交
取消