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

我有这个程序:


package main


import (

    "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 做了什么。


您能帮忙解释一下他们的行为吗?


HUX布斯
浏览 106回答 1
1回答

ABOUTYOU

和 都有f1()接收f2()操作。这些是阻塞操作:只要通道上没有人发送任何内容,它们就会等待。所以你启动f1()并f2()作为新的 goroutine,然后main()睡觉。同时f1()和f2()正在等待来自ch1和的数据ch2。然后main()醒来,并尝试在 上发送一个值ch1。这是可以的,因为有一个 goroutine 准备好从它接收数据 (&nbsp;f1())。然后main()尝试发送ch2,这也可以,已经f2()准备好接收了。然后main()返回,应用程序结束(它不等待其他 goroutine 打印)。如果您不启动f1()并且f2()作为新的 goroutine,当main()到达发送语句时,将没有人准备好从通道接收数据,并且由于它是无缓冲的,因此它将阻塞。由于不会再有任何 goroutine 运行,因此这是一个僵局。
打开App,查看更多内容
随时随地看视频慕课网APP

相关分类

Go