我有這個程序: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")}我預(yù)計,只要 f1 和 f2 不打印任何內(nèi)容,就意味著 ch1 和 ch2 內(nèi)部沒有任何內(nèi)容,因此ch1<-1andch2<-2應(yīng)該阻塞?但運行時,它會打?。簄o buffered channel will wait?main exits為什么那些無緩沖的通道在主通道中ch1沒有ch2被阻塞?如果我不調(diào)用f1/ f2in main,就會報錯dead lock。我不明白 f1/f2 對 ch1/ch2 做了什么。您能幫忙解釋一下他們的行為嗎?
1 回答

ABOUTYOU
TA貢獻1812條經(jīng)驗 獲得超5個贊
和 都有f1()
接收f2()
操作。這些是阻塞操作:只要通道上沒有人發(fā)送任何內(nèi)容,它們就會等待。
所以你啟動f1()
并f2()
作為新的 goroutine,然后main()
睡覺。同時f1()
和f2()
正在等待來自ch1
和的數(shù)據(jù)ch2
。
然后main()
醒來,并嘗試在 上發(fā)送一個值ch1
。這是可以的,因為有一個 goroutine 準備好從它接收數(shù)據(jù) ( f1()
)。然后main()
嘗試發(fā)送ch2
,這也可以,已經(jīng)f2()
準備好接收了。
然后main()
返回,應(yīng)用程序結(jié)束(它不等待其他 goroutine 打?。?/p>
如果您不啟動f1()
并且f2()
作為新的 goroutine,當main()
到達發(fā)送語句時,將沒有人準備好從通道接收數(shù)據(jù),并且由于它是無緩沖的,因此它將阻塞。由于不會再有任何 goroutine 運行,因此這是一個僵局。
- 1 回答
- 0 關(guān)注
- 169 瀏覽
添加回答
舉報
0/150
提交
取消