Golang无限循环超时

标签 go concurrency

我正在尝试读取恒定的数据流,如果接收流的调用时间超过 30 秒,我需要超时并退出程序。我不知道如何在收到超时后退出 go 例程。

func ReceiveStreamMessages(strm Stream, msg chan<- []byte) error {
  d := make(chan []byte, 1)

  e := make(chan error)

  tm := time.After(30 * time.Second)

  go func() {
    for {
        //blocking call
        data, err := strm.Recv()
        if err != nil {
            e <- err
            return
        }
        select {
        case d <- data.Result:
        case <-tm:
            //exit out go routine
            return
         }
      }
  }()

  for {
    select {
    case message := <-d:
        msg <- message
    case err := <-e:
        return err
    case <-tm:
        return nil
    }
  }
}

我上面的代码是错误的:为了让 select 在 go 例程的 for 循环中运行,阻塞函数必须返回并且数据将被填充,因此不会命中超时选择情况(或者随机执行,因为两者都已准备好)。退出父函数是否足以退出 go 例程?

最佳答案

使用contextWithTimeout。像这样的事情:

package main

import (
    "context"
    "fmt"
    "sync"
    "time"
)

func main() {
    ctx, cancel := context.WithTimeout(context.Background(), 1*time.Second)
    // prepare
    ...
    // wait group just for test
    var wg sync.WaitGroup
    wg.Add(1)
    go func() {
        for {
            select {
            case d <- data.Result:
               // do something
            case <-ctx.Done():
                fmt.Println("Done")
                wg.Done()
                return
            }
        }
    }()
    wg.Wait()
    cancel()
    fmt.Println("Hello, playground")
}

您可以在此处查看一个工作示例 https://play.golang.org/p/agi1fimtEkJ

关于Golang无限循环超时,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/64008452/

相关文章:

css - 如何将 css 包含到 go lang 应用程序

go - 将 strconv.ParseInt 与 int 一起使用的惯用方法

C# 异步方法根据第一个完成的任务执行

java - 同时向 HashMap 添加元素

ruby-on-rails - Rails 4 并发数据库索引

java - 设计考虑 - 在一定数量的失败后关闭 ForkJoinPool

go - 更改 map 链接是否安全?

go - 如何在保留测试可比性的同时使用动态误差?

xml - 在Go中使用unmarshal无法访问命名空间的XML属性

go - go-ethereum `bind.NewTransactor()` 的巨大持久内存分配?