为了账号安全,请及时绑定邮箱和手机立即绑定

无缓冲通道是否等待数据?

无缓冲通道是否等待数据?

Go
开心每一天1111 2022-04-26 14:14:40
我有这个程序: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<-1应该ch2<-2阻塞吗?但是在运行时,它会打印:no buffered channel will wait?main exits为什么那些无缓冲的通道ch1并ch2没有在主通道内被阻塞?如果我不调用f1/ f2in main,就会dead lock报错。我不明白 f1/f2 对 ch1/ch2 做了什么。你能帮忙解释一下他们的行为吗?
查看完整描述

1 回答

?
慕尼黑8549860

TA贡献1818条经验 获得超11个赞

两者f1()都有f2()接收操作。这些是阻塞操作:只要没有人在通道上发送任何东西,他们就会等待

所以你启动f1()f2()作为新的 goroutines,然后main()睡觉。同时f1()f2()正在等待来自ch1和的数据ch2

然后main()醒来,并尝试在 上发送一个值ch1。这没关系,因为有一个 goroutine 准备好接收它(f1())。然后main()尝试发送ch2,这也可以,f2()准备好从中接收。

然后main()返回,app 结束(它不等待其他 goroutine 打印)。

如果您不启动f1()并且f2()作为新的 goroutines,当main()到达发送语句时,将没有人准备好从通道接收,并且由于它是无缓冲的,它将阻塞。而且由于不会有更多的 goroutine 运行,这是一个死锁。


查看完整回答
反对 回复 2022-04-26
  • 1 回答
  • 0 关注
  • 111 浏览
慕课专栏
更多

添加回答

举报

0/150
提交
取消
意见反馈 帮助中心 APP下载
官方微信