1
func (s *service) registerMethods() {
    s.method = make(map[string]*methodType)
    for i := 0; i < s.typ.NumMethod(); i++ {
        method := s.typ.Method(i)
        mType := method.Type
        if mType.NumIn() != 3 || mType.NumOut() != 1 {
            continue
        }
        if mType.Out(0) != reflect.TypeOf((*error)(nil)).Elem() {
            continue
        }
        argType, replyType := mType.In(1), mType.In(2)
        if !isExportedOrBuiltinType(argType) || !isExportedOrBuiltinType(replyType) {
            continue
        }
        s.method[method.Name] = &methodType{
            method:    method,
            ArgType:   argType,
            ReplyType: replyType,
        }
        log.Printf("rpc server: register %s.%s\n", s.name, method.Name)
    }
}

reflect.TypeOf((*error)(nil)).Elem()这段代码是什么意思?我知道if mType.Out(0) != reflect.TypeOf((*error)(nil)).Elem()正在尝试确定方法的返回类型是否是错误而不是。但对我来说,reflect.TypeOf((error)(nil))直觉上会做同样的事情,但不幸的是不会。当我尝试编译这段代码时,它说类型错误不是表达式,在这种情况下是什么意思?不reflect.Typeof()接受某种类型的参数?我发现 (*error)(nil) 等价于 *error = nil。在上面的上下文中,我对这个表达感到困惑。

4

1 回答 1

4

TL;博士; reflect.TypeOf((*error)(nil)).Elem()是用于获取reflect.Type接口类型的类型描述符的表达式error。Usingreflect.TypeOf(error(nil)) 不能用于相同的目的(阅读下面的原因)。


reflect.TypeOf((*error)(nil)).Elem()nil通过使用 type 的类型化指针值来实现其目标*error,将其传递给以reflect.TypeOf()获取类型的reflect.Type描述符*error,并用于Type.Elem()获取 的元素(基)类型的类型描述符*error,即error.

reflect.TypeOf()期望一个interface{}值:

func TypeOf(i interface{}) Type

基本上,无论您传递给什么值reflect.TypeOf(),如果它还不是接口值,它将被interface{}隐式包装。如果传递的值已经是接口值,那么存储在其中的具体值将作为interface{}.

因此,如果您尝试将error值传递给它,因为它error是一个接口类型,存储在其中的具体值将被“重新打包”为一个interface{}值。接口类型error不会被保留/转移!

如果你传递一个nil类型的值error,例如error(nil),由于接口值本身是nil,它不包含具体的值和类型,一个nil interface{}值将被传递,这将导致nil reflect.Type返回。引自reflect.TypeOf()

TypeOf 返回表示 i 的动态类型的反射 Type。如果 i 是一个 nil 接口值,TypeOf 返回 nil。

如果你传递一个类型的值*error(可能是一个nil指针),它不是一个接口值,它是一个指针值(一个指向接口的指针)。所以它将被包装在一个interface{}值中,并且存储在其中的具体值将是 type *error。使用Type.Elem()您可以访问指向的类型,即error.

这是使用指向接口的指针有意义的罕见情况之一,实际上是不可避免的。

查看相关问题:

获取基于原始类型的类型的 reflect.Kind

go中的reflect.ValueOf()和Value.Elem()有什么区别?

隐藏 nil 值,了解 golang 为何在这里失败

于 2021-08-20T18:38:10.277 回答