我有3个通道,可以一起获取数据。我的代码第一次运行完美,但是之后我又发送了另一个数据,出现错误:在关闭的通道上发送。在func FillNotCheckedDeliveryCh那一行中,我关闭了通道。如果我不关闭它,应用程序停止并且不继续。
func main() {
receiveBulkIdsCh := make(chan int64, 100)
NotCheckedDeliveryCh := make(chan CommonType.BasicRequestParameters, 100)
ResultCh := make(chan CommonType.MessageStateResult, 100)
var wg sync.WaitGroup
wg.Add(4)
/* Every BulkIds that receive from queue , save in receiveBulkIdsChan channel*/
go func() {
defer wg.Done()
PopQueue(receiveBulkIdsCh)
}()
go func() {
defer wg.Done()
for BulkId := range receiveBulkIdsCh {
FillNotCheckedDeliveryCh(data, NotCheckedDeliveryCh)
}
}()
go func() {
defer wg.Done()
for item := range NotCheckedDeliveryCh {
for msgStatus := range DoFuncGetMessageState(item) {
ResultCh <- msgStatus
}
close(ResultCh)
}
}()
go func() {
defer wg.Done()
for Result := range ResultCh {
ReadResultCh(Result)
}
}()
wg.Wait()
}
func FillNotCheckedDeliveryCh(data IntegrateRowsFields, ch chan<- CommonType.BasicRequestParameters) {
ch <- PackerForNotFinishedCh(data)
defer close(ch)
}
可以帮助解决什么问题吗?
答案 0 :(得分:2)
在Go中使用频道时,发件人始终应关闭频道。因为这表明不再有数据通过该通道发送。
在您的代码中,接收方正在关闭它。只需打开此处的频道并删除其中的close(ch)
。
func FillNotCheckedDeliveryCh(data IntegrateRowsFields, ch chan<- CommonType.BasicRequestParameters) {
ch <- PackerForNotFinishedCh(data)
defer close(ch)
}
如果该部分有障碍物,则可以使用select
:
func FillNotCheckedDeliveryCh(data IntegrateRowsFields, ch chan<- CommonType.BasicRequestParameters) {
select {
case ch <- PackerForNotFinishedCh(data):
default:
}
}
答案 1 :(得分:0)
关闭接收器侧的通道,然后尝试在同一通道上发送会给您带来错误。更好的做法是仅在所有值都在通道上发送后,才在go例程之外关闭该通道。
等待所有执行例程完成后关闭通道。
wg.Wait()
close(ch)
当该通道上没有其他要发送的值时,请始终关闭该通道。
接收者可以通过分配一个频道来测试频道是否已关闭 接收表达式的第二个参数:after
v, ok := <-ch
在您的情况下,您也可以使用ok在接收器端进行检查,以查看通道中是否有更多值或该通道是否已关闭。
频道与文件不同;您通常不需要关闭它们。 仅在必须告知接收者没有接收器时才需要关闭 还有更多的值,例如终止范围循环。
Go playground example用于在不关闭通道并在其上移动时生成错误。
Playground example用于在发送所有值后关闭通道。