尝试同步goroutines时出现死锁错误

时间:2017-03-13 15:32:41

标签: go synchronization goroutine

我正面临一个恼人的问题。当我尝试使用wg.Add()同步我的例程时,会引发死锁错误。

package main

import (
    "fmt"
    "sync"
)

func hello(ch chan int, num int, wg *sync.WaitGroup) {
    for {
        i := <-ch
        if i == num {
            fmt.Println("Hello number:", i)
            ch <- (num - 1)
            defer wg.Done() // Same happens without defer
            return
        }
        ch <- i
    }
}

func main() {
    fmt.Println("Start")

    var wg sync.WaitGroup
    ch := make(chan int)

    for i := 0; i < 10; i++ {
        wg.Add(1)
        go hello(ch, i, &wg)
    }

    ch <- 9

    wg.Wait()

    fmt.Println("End")
}

输出:

Start
Hello number: 9
Hello number: 8
Hello number: 7
Hello number: 6
Hello number: 5
Hello number: 4
Hello number: 3
Hello number: 2
Hello number: 1
Hello number: 0
fatal error: all goroutines are asleep - deadlock!

goroutine 1 [semacquire]:
sync.runtime_Semacquire(0xc04203a20c)
        C:/Go/src/runtime/sema.go:47 +0x3b
sync.(*WaitGroup).Wait(0xc04203a200)
        C:/Go/src/sync/waitgroup.go:131 +0x81
main.main()
        C:/Users/Augusto Dias/Documents/GoLang/MT.go:34 +0x1a0

goroutine 18 [chan send]:
main.hello(0xc0420380c0, 0x0, 0xc04203a200)
        C:/Users/Augusto Dias/Documents/GoLang/MT.go:13 +0x197
created by main.main
        C:/Users/Augusto Dias/Documents/GoLang/MT.go:29 +0x151
exit status 2

当我在for block之外使用wg.Add(9)时,我没有错误。

func main() {
    fmt.Println("Start")

    var wg sync.WaitGroup
    ch := make(chan int)

    wg.Add(9) // Use wg.Add(10) will raise deadlock too

    for i := 0; i < 10; i++ {
        go hello(ch, i, &wg)
    }

    ch <- 9

    wg.Wait()
    fmt.Println("End")
}

输出:

Start
Hello number: 9
Hello number: 8
Hello number: 7
Hello number: 6
Hello number: 5
Hello number: 4
Hello number: 3
Hello number: 2
Hello number: 1
End

为什么会发生这种情况,我的意思是,为什么惯例会在我等待它们时睡着?使用相同的发送和接收通道可能是此问题的根源?

1 个答案:

答案 0 :(得分:3)

频道0(来电go hello(ch, 0, &wg)),因为它是最后一个频道被卡住了这条线

ch <- (num - 1)

它正在尝试发送到某个频道但没有人接收它。因此,该函数将无限期地等待,永远不会完成。

有关如何解决此问题的一些建议

  • 在主循环中创建消费者
  • 使频道ch无阻塞