登录
首页 >  Golang >  Go问答

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

来源:stackoverflow

时间:2024-02-03 21:26:42 398浏览 收藏

知识点掌握了,还需要不断练习才能熟练运用。下面golang学习网给大家带来一个Golang开发实战,手把手教大家学习《为什么下列 go 程序会出现“致命错误:所有 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 仍然可以继续运行。

今天带大家了解了的相关知识,希望对你有所帮助;关于Golang的技术知识我们会一点点深入介绍,欢迎大家关注golang学习网公众号,一起学习编程~

声明:本文转载于:stackoverflow 如有侵犯,请联系study_golang@163.com删除
相关阅读
更多>
最新阅读
更多>
课程推荐
更多>