Golang:在多个goroutine中发送关闭通道错误

时间:2018-08-08 07:32:19

标签: go

我有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)
}

可以帮助解决什么问题吗?

2 个答案:

答案 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用于在发送所有值后关闭通道。