1

任何帮助在这里表示赞赏!我确定我错过了一些非常基本的东西。

我遇到的问题是我试图在演示 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))
}

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

package some_package

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

我知道我遗漏了一些简单的东西,因为如果我尝试上面的代码并将我的getField()函数放在package main一切正常的范围内。

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

4

1 回答 1

2

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

package main

type ctxKey struct{}

不是同一类型

package some_package

type ctxKey struct{}

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

package baz

import (
   "myproject/foo"
   "myproject/some_package"
)

func doBaz() {
    foo := foo.CtxKey{}
    bar := some_package.CtxKey{}
}

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

func (c *valueCtx) Value(key interface{}) interface{} {
    if c.key == key {
        return c.val
    }
    // then it checks if the key exists on the parent
    return c.Context.Value(key)
}

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

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

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

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

package some_ctx_helper_pkg

// unexported, to ensure encapsulation
type ctxKey struct{}

func Set(ctx context.Context, value interface{}) context.Context {
    return context.WithValue(ctx, ctxKey{}, value)
}

func Get(ctx context.Context) interface{} {
    return ctx.Value(ctxKey{})
}
于 2021-04-11T16:10:06.750 回答