在 Go HTTP 处理程序中使用未导出的结构键获取上下文值时,上下文值为 nil

ngr*_*fin 3 httprequest go httprouter go-context

感谢这里的任何帮助!我确信我错过了一些非常基本的东西。

我遇到的问题是我试图在演示 Web 应用程序中获取脱离上下文的值,并且收到错误:

2021/04/11 11:35:54 http: panic serving [::1]:60769: interface conversion: interface {} is nil, not []string

在我的主要函数中,我使用以下内容设置上下文:

package main

type ctxKey struct{}

func someHttpHandleFunc() {
  // .....
  ctx := context.WithValue(r.Context, ctxKey{}, matches[1:])
  route.handle(w, r.WithContext(ctx))
}
Run Code Online (Sandbox Code Playgroud)

然后在我的处理程序中,我有以下内容:

package some_package

type ctxKey struct{}
func getField(r *http.Request, index int) string {
    fields := r.Context().Value(ctxKey{}).([]string)
    return fields[index]
}
Run Code Online (Sandbox Code Playgroud)

我知道我错过了一些简单的东西,因为如果我尝试上面的代码并将我的getField()函数放在package main一切正常的情况下。

作为参考,这是一个学习练习,我正在尝试自学 Go 路由。我确实知道有可用的路由包 - 但我的目标是学习。我正在尽力遵循Go 中 HTTP 路由的不同方法。我还阅读了上下文值的陷阱以及如何在 Go 中避免或减轻它们。后者似乎直接解决了我遇到的问题,但我似乎无法根据现有的情况找出如何解决它。

bla*_*een 7

不同包中定义的结构体类型是不同的

\n
package main\n\ntype ctxKey struct{}\n
Run Code Online (Sandbox Code Playgroud)\n

与以下类型不同

\n
package some_package\n\ntype ctxKey struct{}\n
Run Code Online (Sandbox Code Playgroud)\n

为了使其直观地更清晰,请考虑如果您要从第三个包 \xe2\x80\x94 引用这些类型,让我们假设这些类型已导出 \xe2\x80\x94 您必须导入相应的包并使用适当的选择器:

\n
package baz\n\nimport (\n   "myproject/foo"\n   "myproject/some_package"\n)\n\nfunc doBaz() {\n    foo := foo.CtxKey{}\n    bar := some_package.CtxKey{}\n}\n
Run Code Online (Sandbox Code Playgroud)\n

现在, 的实现Value(key interface{})使用比较运算符==来确定提供的键是否匹配:

\n
func (c *valueCtx) Value(key interface{}) interface{} {\n    if c.key == key {\n        return c.val\n    }\n    // then it checks if the key exists on the parent\n    return c.Context.Value(key)\n}\n
Run Code Online (Sandbox Code Playgroud)\n

这意味着类型也必须匹配。从规范来看,比较运算符

\n
\n

在任何比较中,第一个操作数必须可分配给第二个操作数的类型,反之亦然。

\n
\n

显然,在您的示例中,ctxKey struct{}声明 inmain不可分配给ctxKey struct{}声明 in some_package,反之亦然,因为它们的类型不同。

\n

要解决您的错误,请确保设置和获取上下文值时使用的键是同一类型。最好的方法,也是为了确保正确的封装,可能是从同一个包中设置和获取上下文值:

\n
package some_ctx_helper_pkg\n\n// unexported, to ensure encapsulation\ntype ctxKey struct{}\n\nfunc Set(ctx context.Context, value interface{}) context.Context {\n    return context.WithValue(ctx, ctxKey{}, value)\n}\n\nfunc Get(ctx context.Context) interface{} {\n    return ctx.Value(ctxKey{})\n}\n
Run Code Online (Sandbox Code Playgroud)\n