登录
首页 >  Golang >  Go问答

为什么在同一个 goroutine 中使用无缓冲通道会导致死锁?

来源:Golang技术栈

时间:2023-03-09 20:59:38 158浏览 收藏

今天golang学习网给大家带来了《为什么在同一个 goroutine 中使用无缓冲通道会导致死锁?》,其中涉及到的知识点包括golang等等,无论你是小白还是老手,都适合看一看哦~有好的建议也欢迎大家在评论留言,若是看完有所收获,也希望大家能多多点赞支持呀!一起加油学习~

问题内容

我确信对于这种微不足道的情况有一个简单的解释,但我是go并发模型的新手。

当我运行这个例子时

package main

import "fmt"

func main() {
    c := make(chan int)    
    c 

我收到此错误:

fatal error: all goroutines are asleep - deadlock!

goroutine 1 [chan send]:
main.main()
    /home/tarrsalah/src/go/src/github.com/tarrsalah/tour.golang.org/65.go:8 +0x52
exit status 2

为什么 ?


包装c 在 agoroutine中使示例按我们预期的方式运行

package main

import "fmt"

func main() {
    c := make(chan int)        
    go func(){
       c 

再次,为什么?

拜托,我需要深入的解释,而不仅仅是如何消除死锁和修复代码。

正确答案

文档中

如果通道没有缓冲,发送方会阻塞,直到接收方收到该值。如果通道有缓冲区,发送方只会阻塞,直到值被复制到缓冲区;如果缓冲区已满,这意味着要等到某个接收器检索到一个值。

否则说:

  • 当通道已满时,发送方等待另一个 goroutine 通过接收来腾出一些空间
  • 你可以看到一个无缓冲的通道总是满的:必须有另一个 goroutine 来接收发送者发送的内容。

这条线

c 

阻塞,因为通道没有缓冲。由于没有其他 goroutine 接收该值,情况无法解决,这是一个死锁。

您可以通过将频道创建更改为

c := make(chan int, 1) 

以便在通道阻塞之前为通道中的一个项目留出空间。

但这不是并发的意义所在。通常,您不会使用没有其他 goroutine 的通道来处理您放入的内容。你可以像这样定义一个接收 goroutine:

func main() {
    c := make(chan int)    
    go func() {
        fmt.Println("received:", 

示范

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

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