我有一些代码,我试图使用 fmt.Println 在 goroutine 中打印 channel 的内容。这是代码的简化版本。
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);
}
这里有什么不同?
最佳答案
在第一个版本中, channel 读取是从主 go-routine 执行的 - 因此会出现死锁。 第二个版本,读取是从创建的 go-routine 完成的。
基本上是这样的:
go fmt.Println(<-ch);
变成:
v := <-ch
go fmt.Println(v);
因为函数的参数是在调用之前计算的。
附言defer 语句的行为类似。因此,如果您希望传递给 go-routine 或 defer 语句的值在“运行时”进行评估,请始终使用闭包。
关于go - channel 仅从 Println in go routine 接收抛出死锁异常,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/55790343/