登录
首页 >  Golang >  Go问答

仅从 go 例程中的 Println 接收通道会引发死锁异常

来源:stackoverflow

时间:2024-04-19 23:30:35 434浏览 收藏

最近发现不少小伙伴都对Golang很感兴趣,所以今天继续给大家介绍Golang相关的知识,本文《仅从 go 例程中的 Println 接收通道会引发死锁异常》主要内容涉及到等等知识点,希望能帮到你!当然如果阅读本文时存在不同想法,可以在评论中表达,但是请勿使用过激的措辞~

问题内容

我有一些代码,我试图使用 fmt.println 打印 goroutine 中通道的内容。这是代码的简化版本。

package main

import "fmt"
import "time"

func main() {
  ch := make(chan int)
  go fmt.println(<-ch);
  ch<- 10;
  time.sleep(time.second * 10);
}

当我运行上面的代码时,我收到此错误。

fatal error: all goroutines are asleep - deadlock!

goroutine 1 [chan receive]:
main.main()
        c:/cygwin64/home/vmadhusudana/go/channel.go:9 +0x67
exit status 2

但是当我从函数调用 fmt.println 时它工作正常

package main

import "fmt"
import "time"

func main() {
    ch := make(chan int)
    go func(){fmt.Println(<-ch)}();
    ch<- 10;
    time.Sleep(time.Second * 10);
}

这里有什么不同?


解决方案


在第一个版本中,通道读取是从主 go 例程执行的 - 因此出现了死锁。 第二个版本,读取是从创建的 go-routine 中完成的。

基本上是这样的:

go fmt.println(<-ch);

变成:

v := <-ch
go fmt.println(v);

由于函数的参数在调用之前进行评估。

附注defer 语句的行为类似。因此,如果您希望传递给 go-routine 或 defer 语句的值在“运行时”进行计算,请始终使用闭包。

// receive the value from channel firstly
// then create new goroutine to println
go fmt.Println(<-ch)

// receive channel in new goroutine 
go func(){fmt.Println(<-ch)}()

https://play.golang.org/p/xMyqd-Yr8_a

这将帮助您理解 exec 的顺序。

到这里,我们也就讲完了《仅从 go 例程中的 Println 接收通道会引发死锁异常》的内容了。个人认为,基础知识的学习和巩固,是为了更好的将其运用到项目中,欢迎关注golang学习网公众号,带你了解更多关于的知识点!

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