我有這個程序: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應該阻塞?但運行時,它會打?。簄o 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 關注
- 184 瀏覽
添加回答
舉報
0/150
提交
取消
