Golang 协程同步:如何避免“所有协程休眠 - 死锁”错误?

golang 协程同步:如何避免“所有协程休眠 - 死锁”错误?

致命错误:所有协程休眠 - 死锁!

golang 中,当使用协程同步时,可能会遇到致命错误“all goroutines are asleep - deadlock”。

此错误的原因可能是协程之间的死锁,即它们都在等待对方动作。让我们通过一个代码示例来理解:

func main() {
    a := make(chan bool)
    b := make(chan bool)
    defer close(a)
    defer close(b)
    var wg sync.waitgroup
    wg.add(2)
    go func() {
        for i := 0; i < 10; i += 2 {
            if <-a {
                fmt.println(i)
                b <- true
            }
        }
        wg.done()
    }()
    go func() {
        for i := 1; i < 10; i += 2 {
            if <-b {
                fmt.println(i)
                a <- true
            }
        }
        wg.done()
    }()
    a <- true
    wg.wait()
}

观察代码,可以看到两个协程交替执行,通过通道 a 和 b 进行通信。问题出在 b 协程的最后一次写入 a

为了解决此问题,我们需要添加一个额外的读入操作,如:

    go func() {
        for i := 1; i < 10; i += 2 {
            if <-B {
                fmt.Println(i)
                A <- true
            }
        }
        <-A
        wg.Done()
    }()

这样,在 b 协程结束前,它会等待从 a 通道读入一个值,从而避免死锁。

以上就是Golang 协程同步:如何避免“所有协程休眠 - 死锁”错误?的详细内容,更多请关注其它相关文章!