登录
首页 >  Golang >  Go问答

冷静应对这种情况

来源:stackoverflow

时间:2024-03-26 14:33:31 236浏览 收藏

当使用无缓冲通道时,如果没有另一个 goroutine 同时进行读取,写入操作将导致死锁,从而引发恐慌。在第一个示例中,主 goroutine 试图向通道写入一个值,但没有其他 goroutine 准备接收它。而在第二个示例中,当尝试从通道读取值时,没有 goroutine 准备同时发送值。

问题内容

我看到了恐慌

func main() {
    ch := make(chan int)
    ch <- 1
    fmt.println(<-ch)
}

fatal error: all goroutines are asleep - deadlock!

goroutine 1 [chan send]:
main.main()
    /tmp/sandbox058504389/main.go:10 +0x60

我相信,go 检测到主例程在写入时被阻塞,并且由于在写入无缓冲通道时发生了阻塞,因此永远不会到达读取片段,我们会看到恐慌。为什么我们在下面的代码中看不到这个问题,为什么 go 不恐慌

func main() {
    fmt.Println("Hello, playground")

    var wg sync.WaitGroup

    ch := make(chan int)

    wg.Add(1)
    go func() {
        <-ch 
        wg.Done()
    }()

    wg.Wait()
}

而如果在工作片段中,如果我将 <-ch 即通道读取更改为 ch<-(即写入),我确实会看到错误。有人可以解释一下为什么吗?


解决方案


在这两个示例中,这是通道的行为。有两种类型的通道:缓冲通道和无缓冲通道。

缓冲通道具有一定的容量来存储通道内的项目。就像一个缓冲区。

无缓冲通道没有地方可以在通道中存储项目,这意味着在将任何内容发送到/写入到无缓冲通道之前,需要从其中读取很多内容。这意味着通道的每一端都必须有一个 goroutine,一个准备发送,一个准备接收。

在第一个示例中:

func main() {
    ch := make(chan int)
    ch <- 1
    fmt.println(<-ch)
}

你有 1 个 goroutine(主要的 go 例程)。这个单一 goroutine 尝试向通道 ch <- 1 写入一个值,但是没有其他 goroutine 准备好同时接收该值!函数 fmt.println(<-ch) 但这是在写入之后执行的,但是写入无法成功,因为在写入的同时没有读取就绪。你需要 2 个 goroutine!

在下一个示例中:

func main() {
    var wg sync.WaitGroup

    ch := make(chan int)

    wg.Add(1)
    go func() {
        <-ch 
        wg.Done()
    }()

    wg.Wait()
}

在演示中进行测试时,这也会引起恐慌。这是因为您已经使用 go func() {... } 创建了第二个 goroutine,它尝试从通道读取值,但没有任何东西同时发送值。所以所有的 goroutine 都被阻塞了,并且出现了恐慌。

关于此示例需要注意的一件事是,如果通道 ch 已关闭,并且在读取发生之前使用 close(ch),则读取将始终成功。它将能够从通道读取零值(在本例中为 0)。

正如您所指出的,调整上面的示例以在匿名函数 ch<- 中进行写入也会出现恐慌,因为没有第二个例程准备好同时接收该值!嗯>

这是一个简化的、经过调整的工作版本,显示您需要在无缓冲通道上同时进行读取和写入。

https://play.golang.org/p/gNzgA9Vosm6

到这里,我们也就讲完了《冷静应对这种情况》的内容了。个人认为,基础知识的学习和巩固,是为了更好的将其运用到项目中,欢迎关注golang学习网公众号,带你了解更多关于的知识点!

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