登录
首页 >  Golang >  Go问答

没有取消传播的上下文

来源:stackoverflow

时间:2024-04-08 20:18:35 396浏览 收藏

在Golang实战开发的过程中,我们经常会遇到一些这样那样的问题,然后要卡好半天,等问题解决了才发现原来一些细节知识点还是没有掌握好。今天golang学习网就整理分享《没有取消传播的上下文》,聊聊,希望可以帮助到正在努力赚钱的你。

问题内容

如何创建 go 上下文的副本(如果愿意,可以是克隆),其中包含原始版本中存储的所有值,但不会在原始版本取消时被取消?

对我来说,这确实是一个有效的用例。假设我有一个 http 请求,并且在响应返回到客户端后,其上下文被取消,并且我需要在该请求结束时在一个单独的 goroutine 中运行一个异步任务,该 goroutine 很可能比父上下文的寿命更长。

func Handler(ctx context.Context) (interface{}, error) {
        result := doStuff(ctx)
        newContext := howDoICloneYou(ctx)
        go func() {
                doSomethingElse(newContext)
        }()
        return result
}

任何人都可以建议如何完成此操作吗?

当然,我可以跟踪可能放入上下文中的所有值,创建一个新的背景 ctx,然后迭代每个可能的值并复制...但这看起来很乏味,并且很难在庞大的代码库。


解决方案


更新:go 1.21 将 WithoutCancel 添加到上下文包中。

由于 context.context 是一个接口,因此您可以简单地创建自己的永远不会取消的实现:

import (
    "context"
    "time"
)

type noCancel struct {
    ctx context.Context
}

func (c noCancel) Deadline() (time.Time, bool)       { return time.Time{}, false }
func (c noCancel) Done() <-chan struct{}             { return nil }
func (c noCancel) Err() error                        { return nil }
func (c noCancel) Value(key interface{}) interface{} { return c.ctx.Value(key) }

// WithoutCancel returns a context that is never canceled.
func WithoutCancel(ctx context.Context) context.Context {
    return noCancel{ctx: ctx}
}

是的。不要这样做。

如果您需要不同的上下文,例如为您的异步后台任务创建一个上下文。您传入的上下文和后台任务之一不相关,因此您不得尝试重用传入的上下文。

如果不相关的新上下文需要原始数据中的一些数据:复制您需要的内容并添加新内容。

以上就是《没有取消传播的上下文》的详细内容,更多关于的资料请关注golang学习网公众号!

声明:本文转载于:stackoverflow 如有侵犯,请联系study_golang@163.com删除
相关阅读
更多>
最新阅读
更多>
课程推荐
更多>