如何在不产生死锁的情况下拥有一个缓冲通道和多个读取器?

时间:2016-02-24 12:19:47

标签: go concurrency deadlock channel

  

致命错误所有go例程都处于睡眠状态。死锁。

这就是我尝试过的。我打电话给wg.Done()。缺少什么?

package main

import (
    "fmt"
    "strconv"
    "sync"
)

func sender(wg *sync.WaitGroup, cs chan int) {
    defer wg.Done()
    for i := 0; i < 2; i++ {
        fmt.Println(i)
        cs <- i
    }
}

func reciever(wg *sync.WaitGroup, cs chan int) {
    x, ok := <-cs
    for ok {
        fmt.Println("Retrieved" + strconv.Itoa(x))
        x, ok = <-cs
        if !ok {
            wg.Done()
            break
        }
    }
}

func main() {
    wg := &sync.WaitGroup{}
    cs := make(chan int, 1000)
    wg.Add(1)
    go sender(wg, cs)
    for i := 1; i < 30; i++ {
        wg.Add(1)
        go reciever(wg, cs)
    }
    wg.Wait()
    close(cs)
}

2 个答案:

答案 0 :(得分:1)

您应该在wg.Wait之前关闭频道 所有接收器都在等待来自通道的数据。这就是你陷入僵局的原因 您可以在defer功能sender声明中关闭频道 如果首次尝试从频道接收失败(因为频道已经关闭),您还需要wg.Done()

http://play.golang.org/p/qdEIEfY-kl

答案 1 :(得分:0)

有几件事情:

  1. 发件人完成后,您需要关闭频道。
  2. 在接收器中,通道范围
  3. 不需要在等待组中添加1并在发件人中调用完成
  4. 请参阅http://play.golang.org/p/vz39RY6WA7