首页  >  文章  >  后端开发  >  为什么下面的 go 程序会出现死锁错误“致命错误:所有 goroutine 都在睡眠 - 死锁!”

为什么下面的 go 程序会出现死锁错误“致命错误:所有 goroutine 都在睡眠 - 死锁!”

PHPz
PHPz转载
2024-02-09 08:00:11401浏览

为什么下面的 go 程序会出现死锁错误“致命错误:所有 goroutine 都在睡眠 - 死锁!”

在Go语言中,死锁是一个常见的错误,当所有的goroutine都处于睡眠状态时,就会出现致命错误:"致命错误:所有goroutine都在睡眠 - 死锁!"。这种情况通常发生在多个goroutine之间存在互相等待的情况下。当一个goroutine等待另一个goroutine完成某个操作,而另一个goroutine又在等待第一个goroutine完成某个操作时,就会导致死锁的发生。这种情况下,程序无法继续执行下去,因为所有的goroutine都无法继续执行。为了避免死锁错误的发生,我们需要仔细设计和管理goroutine之间的依赖关系,确保它们能够正确地协同工作。

问题内容

我是 golang 新手,正在尝试使用渠道的一些典型生产者消费者。我知道生产者和消费者都应该从同一个通道写入和读取。但只是为了实验,我让他们从不同的渠道写入和读取,如下所示

package main
    
    import (
        "fmt"
        "sync"
    )
    
    func main() {
    
        var wg sync.WaitGroup
        wg.Add(2)
    
        fmt.Println("Starting the main application")
    
        channel :=make( chan int)
        channel1 :=make( chan int)
    
    
        go generateNumbersV2(&wg, channel)
        go printNumbersV2(&wg, channel1)
    
    
        fmt.Println("Waiting for other goroutines")
        wg.Wait()
        //close()
        //close(channel)
    
        fmt.Println("All goroutines finished")
    }
    
    func printNumbersV2(wg *sync.WaitGroup, rc <-chan int) {
        defer wg.Done()
    
        for idx:=0 ;idx<3;idx++ {
    
            val := <-rc
            fmt.Println("******value received from channel ",val)
        }
    }
    
    func generateNumbersV2(wg *sync.WaitGroup, wc chan<- int) {
        defer wg.Done()
    
        for idx:=0 ;idx<3;idx++ {
    
             wc<-idx
            fmt.Println("###value written to channel ",idx)
        }
    
    }

当我运行该程序时,出现以下错误。

fatal error: all goroutines are asleep - deadlock!

现在,虽然我知道两个 goroutine 都被阻塞,一个在对通道 1 的读取调用上,另一个在对通道的写入调用上被阻塞,因此程序永远不会终止。 但我的问题是,如果实际上正在等待这些通道中的值,程序不应该无限期地等待而不是将其声明为死锁吗?如果稍后由于某些网络读/写,值到达并且其他一些 go 例程写入这些通道,该怎么办?

解决方法

只有当所有 goroutine 都在同步原语上被阻塞时,正在运行的程序才会因死锁恐慌而终止。如果所有 goroutine 都被阻塞等待通道操作和/或互斥锁,则无法进行网络接收,因为没有 goroutine 正在侦听网络连接。这也意味着,在一个有很多 goroutine 的程序中,你可能有很多死锁的 goroutine 组,但是程序仍然继续运行,因为还有其他 goroutine 仍然可以继续运行。

以上是为什么下面的 go 程序会出现死锁错误“致命错误:所有 goroutine 都在睡眠 - 死锁!”的详细内容。更多信息请关注PHP中文网其他相关文章!

声明:
本文转载于:stackoverflow.com。如有侵权,请联系admin@php.cn删除