golang 反射 reflect

reflect 是 golang 的一个标准库, 通过反射可以获取变量的类型、值、tag 等,它是实现 gorm、json、yaml 等库的基础. golang 语言通过反射可以在运行时动态的调用对象的方法和属性.

源码结构

golang reflect 反射的源码位于 golang/go/src/reflect 中,目录下包含以下文件

  • type.go
  • value.go
  • swapper.go
  • deepequal.go
  • makefunc.go
  • all_test.go
  • asm_s390x.s
  • example_test.go
  • set_test.go
  • export_test.go
  • tostring_test.go
  • asm_amd64p32.s
  • asm_mips64x.s
  • asm_386.s
  • asm_arm.s
  • asm_mipsx.s
  • asm_wasm.s
  • asm_amd64.s
  • asm_arm64.s
  • asm_ppc64x.s

在源码中, .s 是一些 golang 汇编文件,核心代码 type.go 3155 行代码, value.go 包含 2774 行代码,swapper.go 及 deepequal.go makefunc.go 总计 309 行代码,剩余 go test 源码 7536 余行.
可以看到 reflect 库的核心源码仅 6000 余行, 对这些代码的理解, 有助于更深刻的理解 golang 的 struct 及一些优秀框架、模块的实现原理.

golang struct

相比与其它语言, golang 的 struct 中可以在类型后增加一段字符串, 即 Tag,在编写程序时,常常通过如下形式对数据进行解析等处理,比如下图所示:

    type jsonStruct struct {
        A int `json:"a,omitempty"`
    }

上面的 struct 中, 变量包含 value、type 及 tag, 其中 tag 内的成员标签变量时 key、value 健值对形式的,参考 StructTag.Lookup(key string) 中获取 tag 中指定 key 的方法

通过 json.Unmarshal 可以将 json 字符串解析到 jsonStruct 类型对象中,同样可以将该类型对象格式化为 json 字符串. 而 json.Unmarshal 及 json.Marshal 都是通过 reflect 实现的.

基础库 reflect

反射是用来检测存储在变量内部(值 value; 类型 concrete type) pair 对的一种机制. golang reflect 提供了两种类型用于访问接口变量的内容, reflect.ValueOf() 和 reflect.TypeOf(),分别用于获取 pair 中的 value 和 type. 示例如下

package main

import (
    "fmt"
    "reflect"
)

func main() {
    var num float64 = 1.2345

    fmt.Println("type: ", reflect.TypeOf(num))
    fmt.Println("value: ", reflect.ValueOf(num))
}

运行结果:
type:  float64
value:  1.2345
  • ValueOf 可用于获取类型的 value
    • Interface()、CanInterface()、Set() …
  • TypeOf 可用于获取类型的 type
    • 通过 TypeOf 可以获取 struct 的结构信息,包括名称、类型、附加 tag
    • StructField
      • Name、Type、Tag
      • Index、Anonymous、PkgPath、Offset
    • Tag/StructTag
      • LookUp
      • Get

优质项目

reflect 解决的是运行时获取的对象方法、属性. 在网络中的数据传输,往往时不携带类型信息的字符串、二进制压缩数据. 那么 golang 程序在与其它进程进行通信时,通过 reflect, 便可以很好的完成数据的解析与序列化. 如远程数据库操作、RPC 数据、http 数据、本地 yaml 文件读取等. 相关的这些优质开源库便是通过反射实现,如下面这些 golang 开发中经常用到的框架:

  • gorm
  • json
  • yaml
  • gRPC
  • protobuf
  • gin.Bind()

gin 中反射的应用

对于刚提到的用了 reflect 的优质开源项目,这里以 gin 框架为例,看看是如何将 reflect 运用上去的. 平时我们写代码时是怎样不知不觉的将反射运用到项目中的.

使用 gin 框架时,经常会通过 Bind() 相关的函数将传入的参数转化为结构体. 下面以 gin.Context.Bind() 为例介绍反射是如何被运用到代码中的.

在 gin-gonic/gin/Context.go 的 Bind() -> MustBindWith() -> ShouldBindWith() -> binding.Bind() 找到 Bind() 方法,它是接口(interface) Binding 的一个方法

type Binding interface {
    Name() string
    Bind(*http.Request, interface{}) error
}

在 gin-gonic/gin/binding 中, 实现上面 Binding 接口 的 struct 有如下:

  • form/formBinding
  • json/jsonBinding
  • msgpack/msgpackBinding
  • protobuf/protobufBinding
  • query/queryBinding
  • xml/xmlBinding

从名称可以看到,它们就是 http 传输过程中常用的参数传输的格式.

其中, gin 框架的 form 在实现 Bind() 方法时候,用到了 mapForm 方法,它位于 binding/form_mapping 中的, 而 mapForm 中使用 reflect 实现了 form 数据到 map[string][]string 的转换, 代码如下:

func mapForm(ptr interface{}, form map[string][]string) error {
    typ := reflect.TypeOf(ptr).Elem()
    val := reflect.ValueOf(ptr).Elem()
    for i := 0; i < typ.NumField(); i++ {
        typeField := typ.Field(i)
        structField := val.Field(i)
        if !structField.CanSet() {
            continue
        }

        structFieldKind := structField.Kind()
        inputFieldName := typeField.Tag.Get("form")
        inputFieldNameList := strings.Split(inputFieldName, ",")
        inputFieldName = inputFieldNameList[0]
        var defaultValue string
        if len(inputFieldNameList) > 1 {
            defaultList := strings.SplitN(inputFieldNameList[1], "=", 2)
            if defaultList[0] == "default" {
                defaultValue = defaultList[1]
            }
        }
        if inputFieldName == "" {
            inputFieldName = typeField.Name

            // if "form" tag is nil, we inspect if the field is a struct or struct pointer.
            // this would not make sense for JSON parsing but it does for a form
            // since data is flatten
            if structFieldKind == reflect.Ptr {
                if !structField.Elem().IsValid() {
                    structField.Set(reflect.New(structField.Type().Elem()))
                }
                structField = structField.Elem()
                structFieldKind = structField.Kind()
            }
            if structFieldKind == reflect.Struct {
                err := mapForm(structField.Addr().Interface(), form)
                if err != nil {
                    return err
                }
                continue
            }
        }
        inputValue, exists := form[inputFieldName]

        if !exists {
            if defaultValue == "" {
                continue
            }
            inputValue = make([]string, 1)
            inputValue[0] = defaultValue
        }

        numElems := len(inputValue)
        if structFieldKind == reflect.Slice && numElems > 0 {
            sliceOf := structField.Type().Elem().Kind()
            slice := reflect.MakeSlice(structField.Type(), numElems, numElems)
            for i := 0; i < numElems; i++ {
                if err := setWithProperType(sliceOf, inputValue[i], slice.Index(i)); err != nil {
                    return err
                }
            }
            val.Field(i).Set(slice)
            continue
        }
        if _, isTime := structField.Interface().(time.Time); isTime {
            if err := setTimeField(inputValue[0], typeField, structField); err != nil {
                return err
            }
            continue
        }
        if err := setWithProperType(typeField.Type.Kind(), inputValue[0], structField); err != nil {
            return err
        }
    }
    return nil
}

对于, json、msgpack、protobuf、xml 等格式的转化,则直接继续调用相关库进行 decode 完成. 具体应用反射的过程,可以进一步查看它们的源码.

缺陷

  1. 性能问题
    • 涉及内存分配及后续 GC . 每一次的反射运用,都会涉及到 GC. golang GC 是十分低效的,可能会导致不可预料的性能效率问题.
    • reflect 里面有大量枚举, for 循环
  2. 代码可读性. 过多的运用反射, 在运行时才通过反射确定类型,可能导致后续维护时困难.

总结

在项目中,可能很少会直接使用到 reflect, 但是反射 reflect 作为 golang 40 余个基础库其中之一,使用是及其广泛的,尤其是 golang 的 struct 中经常会用到 Tag 类型,它对应到底层便会使用反射. 另外, 许多优秀的开源程序都会在底层用到 reflect,如上文提到的 gin、gorm、gRPC 等.

理解底层的 reflect, 可以加深对 golang 的结构体 struct 的理解. 在做一些程序设计时,也可在脑海里多提供一种解决问题的思路. 对于各个库之间的关系,也可以更加清晰,有利于梳理技术栈.

开源的框架、组件可能有几百、上千个,但是基础库只有 40 多个,它们反而是实现框架、组建、程序等实现的基础. 掌握 golang 的基础库,再阅读优秀的 golang 源码,可以更快速的理解.

ref