我如何创建一个 Go 上下文的副本(如果你愿意的话,一个克隆),它包含存储在原始文件中的所有值,但不会在原始文件被取消时被取消?
对我来说,这确实是一个有效的用例。假设我有一个 http 请求,它的上下文在响应返回给客户端后被取消,我需要在这个请求结束时在一个单独的 goroutine 中运行一个异步任务,这个 goroutine 很可能比父上下文活得更久。
func Handler(ctx context.Context) (interface{}, error) {
result := doStuff(ctx)
newContext := howDoICloneYou(ctx)
go func() {
doSomethingElse(newContext)
}()
return result
}
有人可以建议应该如何完成吗?
当然我可以跟踪所有可能放入上下文的值,创建一个新的背景 ctx 然后遍历每个可能的值并复制......但这看起来很乏味并且很难在大型代码库。
最佳答案
由于 context.Context 是一个接口(interface),您可以简单地创建您自己的永不取消的实现:
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}
}
关于go - 没有取消传播的上下文,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/54128834/