[笔记]为什么不能在struct中存Contexts?

yujie_7be3 · 2019-04-10 01:34:43 · 1182 次点击 · 预计阅读时间 2 分钟 · 大约8小时之前 开始浏览    
这是一个创建于 2019-04-10 01:34:43 的文章,其中的信息可能已经有所发展或是发生改变。

引用来自 golang team 成员 Sajmani 的 issues 回答

Yes, the impedance mismatch is primarily a concern for exported types. The scenario I'm worried about is:

  1. package foo defines type Context that contains a context.Context and, say, method Foo() *Foo, for accessing foo-specific data.
  2. package bar defines type Context that contains a context.Context and, say, method Bar() *Bar, for accessing bar-specific data.
  3. Some function that accepts a foo.Context needs to call a bar.Context. Passing just the context.Context loses the *Foo and doesn't necessarily know what to pass for *Bar:
func F(fctx foo.Context) {
 bctx := bar.NewContext(fctx.Context(), nil /*the *Bar value*/)
 G(bctx)
}

Furthermore, if G later calls a function that expects a foo.Context, the *Foo that should have been plumbed through has been lost:

func G(bctx bar.Context) {
 fctx := foo.NewContext(bctx.Context(), nil /*the *Foo value*/)
 H(fctx)
}

The point of the context.Context.Value design is that packages foo and bar don't care about any of this. Their values propagate through layers of the call stack without intersecting or interfering with each other. This is exactly what you want for things like trace IDs, authentication scopes, profiling tags. But this is not what you should use for API-visible options. The "keep Context out of structs" restriction is a simple rule to follow that helps users avoid problems like this, but it is overly restrictive. The real guidelines are harder to articulate, but perhaps we should try harder to do so.

我个人的理解是:

  1. context.Context设计的目的是简明描述调用栈中各层级间的关系,传参是达到这一目的的一种简单方式。
  2. 不在struct中存 context.Context只是一种为实现上述目的的编码约束,并不涉及运行机制。
  3. 所以,其实只要自己愿意,也是可以违反的。

有疑问加站长微信联系(非本文作者)

本文来自:简书

感谢作者:yujie_7be3

查看原文:[笔记]为什么不能在struct中存Contexts?

入群交流(和以上内容无关):加入Go大咖交流群,或添加微信:liuxiaoyan-s 备注:入群;或加QQ群:692541889

1182 次点击  
加入收藏 微博
暂无回复
添加一条新回复 (您需要 登录 后才能回复 没有账号 ?)
  • 请尽量让自己的回复能够对别人有帮助
  • 支持 Markdown 格式, **粗体**、~~删除线~~、`单行代码`
  • 支持 @ 本站用户;支持表情(输入 : 提示),见 Emoji cheat sheet
  • 图片支持拖拽、截图粘贴等方式上传