文章详情

短信预约-IT技能 免费直播动态提醒

请输入下面的图形验证码

提交验证

短信预约提醒成功

goroutine 没有看到上下文取消?

2024-04-04 23:47

关注

小伙伴们对Golang编程感兴趣吗?是否正在学习相关知识点?如果是,那么本文《goroutine 没有看到上下文取消?》,就很适合你,本篇文章讲解的知识点主要包括。在之后的文章中也会多多分享相关知识点,希望对大家的知识积累有所帮助!

问题内容

我有两个 goroutine 同时运行。

在某些时候,我希望我的程序能够正常退出,因此我使用 cancel() 函数来通知我的 goroutine 它们需要停止,但两者中只有一个收到消息。

这是我的主要内容(简化):

ctx := context.background()
ctx, cancel := context.withcancel(ctx)

done := make(chan os.signal, 1)
signal.notify(done, os.interrupt, syscall.sigint, syscall.sigterm)

wg := &sync.waitgroup{}
wg.add(2)

go func() {
    err := eng.watcher(ctx, wg)
    if err != nil {
        cancel()
    }
}()

go func() {
    err := eng.suspender(ctx, wg)
    if err != nil {
        cancel()
    }
}()

<-done // wait for sigint / sigterm
log.print("receive shutdown")
cancel()
wg.wait()

log.print("controller exited properly")

suspender goroutine 成功存在(代码如下):

package main

import (
    "context"
    "sync"
    "time"

    log "github.com/sirupsen/logrus"
    metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
    "k8s.io/client-go/util/retry"
)

func (eng *engine) suspender(ctx context.context, wg *sync.waitgroup) error {

    contextlogger := eng.logger.withfields(log.fields{
        "go-routine": "suspender",
    })
    contextlogger.info("starting suspender goroutine")
    now := time.now().in(eng.loc)

    for {
        select {
        case n := <-eng.wl:
            //dostuff


        case <-ctx.done():
            // the context is over, stop processing results
            contextlogger.infof("goroutine suspender canceled by context")
            return nil
        }
    }

}

这是未接收上下文取消的函数:

package main

import (
    "context"
    "sync"
    "time"

    log "github.com/sirupsen/logrus"
)

func (eng *Engine) Watcher(ctx context.Context, wg *sync.WaitGroup) error {
    contextLogger := eng.logger.WithFields(log.Fields{
        "go-routine":      "Watcher",
        "uptime-schedule": eng.upTimeSchedule,
    })
    contextLogger.Info("starting Watcher goroutine")

    ticker := time.NewTicker(time.Second * 30)
    for {
        select {
        case <-ctx.Done():
            contextLogger.Infof("goroutine watcher canceled by context")
            log.Printf("toto")
            return nil
        case <-ticker.C:
            
                //dostuff
            }
        }
    }
}

你能帮我吗?

谢谢:)


正确答案


你用 errgroup 尝试过吗?它内置了上下文取消功能:

ctx := context.Background()
ctx, cancel := context.WithCancel(ctx)
defer cancel()

done := make(chan os.Signal, 1)
signal.Notify(done, os.Interrupt, syscall.SIGINT, syscall.SIGTERM)

// "golang.org/x/sync/errgroup"
wg, ctx := errgroup.WithContext(ctx)

wg.Go(func() error {
    return eng.Watcher(ctx, wg)
})

wg.Go(func() error {
    return eng.Suspender(ctx, wg)
})

wg.Go(func() error {
    defer cancel()
    <-done
    return nil
})

err := wg.Wait()
if err != nil {
    log.Print(err)
}

log.Print("receive shutdown")
log.Print("controller exited properly")

本篇关于《goroutine 没有看到上下文取消?》的介绍就到此结束啦,但是学无止境,想要了解学习更多关于Golang的相关知识,请关注编程网公众号!

阅读原文内容投诉

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

软考中级精品资料免费领

  • 历年真题答案解析
  • 备考技巧名师总结
  • 高频考点精准押题
  • 2024年上半年信息系统项目管理师第二批次真题及答案解析(完整版)

    难度     813人已做
    查看
  • 【考后总结】2024年5月26日信息系统项目管理师第2批次考情分析

    难度     354人已做
    查看
  • 【考后总结】2024年5月25日信息系统项目管理师第1批次考情分析

    难度     318人已做
    查看
  • 2024年上半年软考高项第一、二批次真题考点汇总(完整版)

    难度     435人已做
    查看
  • 2024年上半年系统架构设计师考试综合知识真题

    难度     224人已做
    查看

相关文章

发现更多好内容

猜你喜欢

AI推送时光机
位置:首页-资讯-后端开发
咦!没有更多了?去看看其它编程学习网 内容吧
首页课程
资料下载
问答资讯