Golang不知道大家是否熟悉?今天我将给大家介绍《当通道以裸返回方式返回时,该通道为零》,这篇文章主要会讲到等等知识点,如果你在看完本篇文章后,有更好的建议或者发现哪里有问题,希望大家都能积极评论指出,谢谢!希望我们能一起加油进步!
我花了相当多的时间尝试调试为什么我的频道不接受任何内容。我设法将问题本地化为与裸返回值返回时命名返回值的范围相关的问题。下面的代码显示了这个问题。
package main
import (
"log"
"sync"
)
var receiver chan int
func setup() (receiver chan int) {
receiver = make(chan int)
return
}
//func setup() (chan int) {
// receiver = make(chan int)
// return receiver
//}
func launch(j int){
for i := 0; i < j; i++ {
receiver <- i
}
}
func main() {
var wg sync.waitgroup
wg.add(10)
rcvr := setup()
go func() {
for r := range rcvr {
log.println(r)
wg.done()
}
}()
launch(10)
wg.wait()
}
运行此代码会产生下一个错误
fatal error: all goroutines are asleep - deadlock!
goroutine 1 [chan send (nil chan)]:
...
我不明白为什么通道应该为零。我尝试使用原始值进行类似的分配,它返回了我所期望的结果。为什么频道为零?这里有什么阴影?
解决方案
编辑:到您编辑的代码:
现在您正在使用 2 个完全不同的频道。您的 main()
中有一个通道,您尝试从中接收信号(使用 作为 range
)。并且您还有另一个通道(全局 receiver
变量,您从未为其分配任何值,因此它仍然是 nil
。并且在 nil
通道上发送值将永远阻塞(有关详细信息,请参阅 How does a non initialized channel behave?)。
所以你已经看到了:launch()
在 nil
通道上发送时被阻止,并且你的 main()
goroutine 等待它启动的另一个 goroutine,该 goroutine 等待 rcvr
通道上没有人发送的值任何东西。
一个简单的方法是将通道传递给您在 main()
中创建的 launch()
,并摆脱全局变量:
func launch(j int, receiver chan int) {
for i := 0; i < j; i++ {
receiver <- i
}
}
调用时:
launch(10, rcvr)
然后它就会工作,输出将是(在Go Playground上尝试):
2009/11/10 23:00:00 0
2009/11/10 23:00:00 1
2009/11/10 23:00:00 2
2009/11/10 23:00:00 3
2009/11/10 23:00:00 4
2009/11/10 23:00:00 5
2009/11/10 23:00:00 6
2009/11/10 23:00:00 7
2009/11/10 23:00:00 8
2009/11/10 23:00:00 9
原答案如下:
问题是您在存储 setup()
返回的通道时在 main 中使用 short variable declaration:
receiver := setup()
这将不会为您的全局 receiver
变量赋值,而是创建一个新的局部变量来隐藏全局变量,并且您的 main()
函数将使用此局部变量。但是您的 launch()
函数将使用全局变量(它看不到 main
的局部变量)。
使用简单的 assignment,一切都会好的:
receiver = Setup()
本篇关于《当通道以裸返回方式返回时,该通道为零》的介绍就到此结束啦,但是学无止境,想要了解学习更多关于Golang的相关知识,请关注编程网公众号!