dran0703 2018-01-15 22:10
浏览 101
已采纳

golang for带缓冲通道的循环

I'm experimenting with golang channels and have an issue where the simple program below does not terminate.

Essentially I want to make some async HTTP get requests and then wait till they are all finished. I'm using a buffered channel but I'm not sure this is the idiomatic way.

func GetPrice(quotes chan string) {
    client := &http.Client{}
    req, _ := http.NewRequest("GET", "https://some/api", nil)
    req.Header.Set("Accept", "application/json")
    res, err := client.Do(req)
    if err != nil {
        panic(err)
    }
    defer res.Body.Close()
    body, err := ioutil.ReadAll(res.Body)
    quotes <- string(body)
}

func main() {
    const max = 3
    quotes := make(chan string, max)
    for i := 0; i < max; i++ {
        go GetPrice(quotes)
    }

    for n := range quotes {
        fmt.Printf("
%s", n)
    }
}

The program successfully prints 3(max) items

{"price":"1.00"}
{"price":"2.00"}
{"price":"3.00"}

but then blocks and never exits.

  • 写回答

4条回答 默认 最新

  • doubi9999 2018-01-16 12:59
    关注

    sync.WaitGroup can be used here to wait for all goroutines and then closing the quotes channel:

    func getPrice(quotes chan<- string, onExit func()) {
        go func() {
            defer onExit()
            client := &http.Client{}
            req, _ := http.NewRequest("GET", "https://some/api", nil)
            req.Header.Set("Accept", "application/json")
            res, err := client.Do(req)
            if err != nil {
                panic(err) // should be handled properly
            }
            defer res.Body.Close()
            body, err := ioutil.ReadAll(res.Body)
            quotes <- string(body)
        }()
    }
    
    func main() {
        var wg sync.WaitGroup
        const max = 3
        quotes := make(chan string, max)
        for i := 0; i < max; i++ {
            wg.Add(1)
            getPrice(quotes, func() { wg.Done() })
        }
    
        go func() {
            defer close(quotes)
            wg.Wait()
        }()
    
        for n := range quotes {
            fmt.Printf("
    %s", n)
        }
    }
    
    本回答被题主选为最佳回答 , 对您是否有帮助呢?
    评论
查看更多回答(3条)

报告相同问题?

悬赏问题

  • ¥15 有了解d3和topogram.js库的吗?有偿请教
  • ¥100 任意维数的K均值聚类
  • ¥15 stamps做sbas-insar,时序沉降图怎么画
  • ¥15 unity第一人称射击小游戏,有demo,在原脚本的基础上进行修改以达到要求
  • ¥15 买了个传感器,根据商家发的代码和步骤使用但是代码报错了不会改,有没有人可以看看
  • ¥15 关于#Java#的问题,如何解决?
  • ¥15 加热介质是液体,换热器壳侧导热系数和总的导热系数怎么算
  • ¥100 嵌入式系统基于PIC16F882和热敏电阻的数字温度计
  • ¥15 cmd cl 0x000007b
  • ¥20 BAPI_PR_CHANGE how to add account assignment information for service line