亚洲在线久爱草,狠狠天天香蕉网,天天搞日日干久草,伊人亚洲日本欧美

為了賬號安全,請及時綁定郵箱和手機立即綁定
已解決430363個問題,去搜搜看,總會有你想問的

go教程選擇語句

go教程選擇語句

Go
江戶川亂折騰 2021-12-07 14:57:40
我正在處理 tour.golang.org 上的示例,但遇到了這段我不太明白的代碼:package mainimport "fmt"func fibonacci(c, quit chan int) {    x, y := 0, 1    for {        select {        case c <- x: // case: send x to channel c?            x, y = y, x+y        case <-quit: // case: receive from channel quit?            fmt.Println("quit")            return        }    }}func main() {    c := make(chan int)    quit := make(chan int)    go func() { // when does this get called?        for i := 0; i < 10; i++ {            fmt.Println(<-c)        }        quit <- 0    }()    fibonacci(c, quit)}我了解通道如何工作的基礎知識,但我不明白上面的 select 語句是如何工作的。教程上的解釋說:“select 語句讓 goroutine 等待多個通信操作。select 阻塞,直到它的一個 case 可以運行,然后它執行那個 case。如果多個 case 準備好了,它會隨機選擇一個。”但是這些案件是如何執行的呢?據我所知,他們說:案例:將 x 發送到通道 c案例:從退出接收我想我明白第二個只有在退出有一個值時才執行,這是稍后在 go func() 中完成的。但是第一個案例檢查的是什么?此外,在 go func() 中,我們顯然是從 c 打印值,但此時 c 不應該有任何內容嗎?我能想到的唯一解釋是 go func() 在調用 fibonacci() 之后以某種方式執行。我猜這是一個我也不完全理解的 goroutine,它看起來很神奇。如果有人可以通過此代碼并告訴我它在做什么,我將不勝感激。
查看完整描述

3 回答

?
守著一只汪

TA貢獻1872條經驗 獲得超4個贊

請記住,通道會阻塞,因此 select 語句如下:


select {

case c <- x: // if I can send to c

    // update my variables

    x, y = y, x+y

case <-quit: // If I can receive from quit then I'm supposed to exit

    fmt.Println("quit")

    return

}

沒有default案例意味著“如果我不能發送到 c 并且我不能從退出中讀取,直到我可以為止?!?/p>


然后在你的主進程中,你分離出另一個讀取c結果的函數


for i:=0; i<10; i++ {

    fmt.Println(<-c)  // read in from c

}

quit <- 0  // send to quit to kill the main process.

這里的關鍵是要記住通道會阻塞,并且您正在使用兩個無緩沖通道。使用go分拆第二個功能可以讓您消費,c所以fibonacci將繼續。


Goroutines 是所謂的“綠色線程”。使用關鍵字啟動函數調用go會將其分拆為一個獨立于主執行線運行的新進程。從本質上說,main()和go func() ...同時運行!這很重要,因為我們在此代碼中使用了生產者/消費者模式。


fibonacci產生值并將它們發送到c,從 main 產生的匿名 goroutine 消耗來自的值c并處理它們(在這種情況下,“處理它們”只是意味著打印到屏幕上)。我們不能簡單地產生所有的值然后消費它們,因為c會阻塞。此外fibonacci將永遠產生更多的值(或直到整數溢出),因此即使您有一個具有無限長緩沖區的魔術通道,它也永遠不會到達消費者。


查看完整回答
反對 回復 2021-12-07
?
莫回無

TA貢獻1865條經驗 獲得超7個贊

理解這個代碼示例有兩個關鍵點:

首先,讓我們回顧一下無緩沖通道的工作原理。從文檔

如果通道未緩沖,則發送方會阻塞,直到接收方收到該值。

請注意,在代碼示例兩個通道,c并且quit是無緩沖。

其次,當我們使用go關鍵字啟動一個新的 goroutine 時,執行將與其他例程并行發生。因此,在本例中,我們同時運行兩個走程序:常規的啟動func main(),并開始了例行go func()...func main()。

我在這里添加了一些內聯注釋,這應該會使事情更清楚: package main import "fmt"

func fibonacci(c, quit chan int) {

    x, y := 0, 1

    for { // this is equivalent to a while loop, without a stop condition

        select {

        case c <- x: // when we can send to channel c, and because c is unbuffered, we can only send to channel c when someone tries to receive from it

            x, y = y, x+y

        case <-quit: // when we can receive from channel quit, and because quit is unbuffered, we can only receive from channel quit when someone tries to send to it

            fmt.Println("quit")

            return

        }

    }

}


func main() {

    c := make(chan int)

    quit := make(chan int)

    go func() { // this runs in another goroutine, separate from the main goroutine

        for i := 0; i < 10; i++ {

            fmt.Println(<-c)

        }

        quit <- 0

    }()

    fibonacci(c, quit) // this doesn't start with the go keyword, so it will run on the go routine started by func main()

}


查看完整回答
反對 回復 2021-12-07
?
楊魅力

TA貢獻1811條經驗 獲得超6個贊

你已經明白了。

在 go func() 中,我們顯然是從 c 打印值,但是 c 中不應該有任何內容嗎?我能想到的唯一解釋是 go func() 在調用 fibonacci() 之后以某種方式執行。我猜這是一個 goroutine

是的, go 關鍵字啟動了一個 goroutine,所以func()將與fibonacci(c, quit) 同時運行。從 Println 中的通道接收只是阻塞,直到有東西要接收


查看完整回答
反對 回復 2021-12-07
  • 3 回答
  • 0 關注
  • 195 瀏覽
慕課專欄
更多

添加回答

舉報

0/150
提交
取消
微信客服

購課補貼
聯系客服咨詢優惠詳情

幫助反饋 APP下載

慕課網APP
您的移動學習伙伴

公眾號

掃描二維碼
關注慕課網微信公眾號