反射是指在程序运行期对程序本身进行访问和修改的能力。程序在编译时,变量被转换为内存地址,变量名不会被编译器写入到可执行部分。在运行程序时,程序无法获取自身的信息。

支持反射的语言可以在程序编译期将变量的反射信息,如字段名称、类型信息、结构体信息等整合到可执行文件中,并给程序提供接口访问反射信息,这样就可以在程序运行期获取类型的反射信息,并且有能力修改它们。

Go程序在运行期使用reflect包访问程序的反射信息。

reflect包

在Go语言的反射机制中,任何接口值都由是一个具体类型具体类型的值两部分组成的。 在Go语言中反射的相关功能由内置的reflect包提供,任意接口值在反射中都可以理解为由reflect.Typereflect.Value两部分组成,并且reflect包提供了reflect.TypeOfreflect.ValueOf两个函数来获取任意对象的Value和Type。

Typeof

在Go语言中,使用reflect.TypeOf()函数可以获得任意值的类型对象(reflect.Type),程序通过类型对象可以访问任意值的类型信息。

  1. package main
  2. import (
  3. "fmt"
  4. "reflect"
  5. )
  6. func reflectType(x interface{}) {
  7. v := reflect.TypeOf(x)
  8. fmt.Printf("%T,%v\n", v, v)
  9. }
  10. func main() {
  11. a := 10
  12. reflectType(a) //*reflect.rtype,int
  13. b := float64(1.234)
  14. reflectType(b) //*reflect.rtype,float64
  15. }

type kind和type name

在反射中关于类型还划分为两种:类型(Type)种类(Kind)。因为在Go语言中我们可以使用type关键字构造很多自定义类型,而种类(Kind)就是指底层的类型,但在反射中,当需要区分指针、结构体等大品种的类型时,就会用到种类(Kind)。 举个例子,我们定义了两个指针类型和两个结构体类型,通过反射查看它们的类型和种类。

  1. package main
  2. import (
  3. "fmt"
  4. "reflect"
  5. )
  6. type cat struct {
  7. name string
  8. }
  9. type dog struct {
  10. name string
  11. }
  12. func reflectType(x interface{}) {
  13. v := reflect.TypeOf(x)
  14. fmt.Printf("类型名:%s,类型的种类:%s\n", v.Name(), v.Kind())
  15. }
  16. func main() {
  17. c := cat{
  18. name: "咖啡",
  19. }
  20. reflectType(c) // *类型名:cat,类型的种类:struct
  21. d := dog{
  22. name: "大黄",
  23. }
  24. reflectType(d) // 类型名:dog,类型的种类:struct
  25. }

Go语言的反射中像数组、切片、Map、指针等类型的变量,它们的.Name()都是返回

reflect包中定义的Kind类型如下:

  1. type Kind uint
  2. const (
  3. Invalid Kind = iota // 非法类型
  4. Bool // 布尔型
  5. Int // 有符号整型
  6. Int8 // 有符号8位整型
  7. Int16 // 有符号16位整型
  8. Int32 // 有符号32位整型
  9. Int64 // 有符号64位整型
  10. Uint // 无符号整型
  11. Uint8 // 无符号8位整型
  12. Uint16 // 无符号16位整型
  13. Uint32 // 无符号32位整型
  14. Uint64 // 无符号64位整型
  15. Uintptr // 指针
  16. Float32 // 单精度浮点数
  17. Float64 // 双精度浮点数
  18. Complex64 // 64位复数类型
  19. Complex128 // 128位复数类型
  20. Array // 数组
  21. Chan // 通道
  22. Func // 函数
  23. Interface // 接口
  24. Map // 映射
  25. Ptr // 指针
  26. Slice // 切片
  27. String // 字符串
  28. Struct // 结构体
  29. UnsafePointer // 底层指针
  30. )

Valueof

reflect.ValueOf()返回的是reflect.Value类型,其中包含了原始值的值信息。reflect.Value与原始值之间可以互相转换。

reflect.Value类型提供的获取原始值的方法如下:

方法 说明
Interface() interface {} 将值以 interface{} 类型返回,可以通过类型断言转换为指定类型
Int() int64 将值以 int 类型返回,所有有符号整型均可以此方式返回
Uint() uint64 将值以 uint 类型返回,所有无符号整型均可以此方式返回
Float() float64 将值以双精度(float64)类型返回,所有浮点数(float32、float64)均可以此方式返回
Bool() bool 将值以 bool 类型返回
Bytes() []bytes 将值以字节数组 []bytes 类型返回
String() string 将值以字符串类型返回

通过反射获取值

  1. func reflectValue(x interface{}) {
  2. v := reflect.ValueOf(x)
  3. k := v.Kind()
  4. switch k {
  5. case reflect.Int64:
  6. // v.Int()从反射中获取整型的原始值,然后通过int64()强制类型转换
  7. fmt.Printf("type is int64, value is %d\n", int64(v.Int()))
  8. case reflect.Float32:
  9. // v.Float()从反射中获取浮点型的原始值,然后通过float32()强制类型转换
  10. fmt.Printf("type is float32, value is %f\n", float32(v.Float()))
  11. case reflect.Float64:
  12. // v.Float()从反射中获取浮点型的原始值,然后通过float64()强制类型转换
  13. fmt.Printf("type is float64, value is %f\n", float64(v.Float()))
  14. }
  15. }
  16. func main() {
  17. var a float32 = 3.14
  18. var b int64 = 100
  19. reflectValue(a) // type is float32, value is 3.140000
  20. reflectValue(b) // type is int64, value is 100
  21. // 将int类型的原始值转换为reflect.Value类型
  22. c := reflect.ValueOf(10)
  23. fmt.Printf("type c :%T\n", c) // type c :reflect.Value
  24. }

通过反射设置值

想要在函数中通过反射修改变量的值,需要注意函数参数传递的是值拷贝,必须传递变量地址才能修改变量值。而反射中使用专有的Elem()方法来获取指针对应的值。

  1. package main
  2. import (
  3. "fmt"
  4. "reflect"
  5. )
  6. func reflectSetValue1(x interface{}) {
  7. v := reflect.ValueOf(x)
  8. if v.Kind() == reflect.Int64 {
  9. v.SetInt(200) //修改的是副本,reflect包会引发panic
  10. }
  11. }
  12. func reflectSetValue2(x interface{}) {
  13. v := reflect.ValueOf(x)
  14. // 反射中使用 Elem()方法获取指针对应的值
  15. if v.Elem().Kind() == reflect.Int64 {
  16. v.Elem().SetInt(200)
  17. }
  18. }
  19. func main() {
  20. var a int64 = 100
  21. // reflectSetValue1(a) //panic: reflect: reflect.Value.SetInt using unaddressable value
  22. reflectSetValue2(&a)
  23. fmt.Println(a)
  24. }

isNil()和isValid()

isNil()

  1. func (v Value) IsNil() bool

IsNil()报告v持有的值是否为nil。v持有的值的分类必须是通道、函数、接口、映射、指针、切片之一;否则IsNil函数会导致panic。

isValid()

  1. func (v Value) IsValid() bool

IsValid()返回v是否持有一个值。如果v是Value零值会返回假,此时v除了IsValid、String、Kind之外的方法都会导致panic。

例子:

IsNil()常被用于判断指针是否为空;IsValid()常被用于判定返回值是否有效。

  1. func main() {
  2. // *int类型空指针
  3. var a *int
  4. fmt.Println("var a *int IsNil:", reflect.ValueOf(a).IsNil())
  5. // nil值
  6. fmt.Println("nil IsValid:", reflect.ValueOf(nil).IsValid())
  7. // 实例化一个匿名结构体
  8. b := struct{}{}
  9. // 尝试从结构体中查找"abc"字段
  10. fmt.Println("不存在的结构体成员:", reflect.ValueOf(b).FieldByName("abc").IsValid())
  11. // 尝试从结构体中查找"abc"方法
  12. fmt.Println("不存在的结构体方法:", reflect.ValueOf(b).MethodByName("abc").IsValid())
  13. // map
  14. c := map[string]int{}
  15. // 尝试从map中查找一个不存在的键
  16. fmt.Println("map中不存在的键:", reflect.ValueOf(c).MapIndex(reflect.ValueOf("嘿嘿")).IsValid())
  17. }

结构体反射

与结构体相关的方法

任意值通过reflect.TypeOf()获得反射对象信息后,如果它的类型是结构体,可以通过反射值对象(reflect.Type)的NumField()Field()方法获得结构体成员的详细信息。

reflect.Type中与获取结构体成员相关的的方法如下表所示。

方法 说明
Field(i int) StructField 根据索引,返回索引对应的结构体字段的信息。
NumField() int 返回结构体成员字段数量。
FieldByName(name string) (StructField, bool) 根据给定字符串返回字符串对应的结构体字段的信息。
FieldByIndex(index []int) StructField 多层成员访问时,根据 []int 提供的每个结构体的字段索引,返回字段的信息。
FieldByNameFunc(match func(string) bool) (StructField,bool) 根据传入的匹配函数匹配需要的字段。
NumMethod() int 返回该类型的方法集中方法的数目
Method(int) Method 返回该类型方法集中的第i个方法
MethodByName(string)(Method, bool) 根据方法名返回该类型方法集中的方法

StructField类型

StructField类型用来描述结构体中的一个字段的信息。

StructField的定义如下:

  1. type StructField struct {
  2. // Name是字段的名字。PkgPath是非导出字段的包路径,对导出字段该字段为""。
  3. // 参见http://golang.org/ref/spec#Uniqueness_of_identifiers
  4. Name string
  5. PkgPath string
  6. Type Type // 字段的类型
  7. Tag StructTag // 字段的标签
  8. Offset uintptr // 字段在结构体中的字节偏移量
  9. Index []int // 用于Type.FieldByIndex时的索引切片
  10. Anonymous bool // 是否匿名字段
  11. }

结构体反射示例

当我们使用反射得到一个结构体数据之后可以通过索引依次获取其字段信息,也可以通过字段名去获取指定的字段信息。

  1. type student struct {
  2. Name string `json:"name"`
  3. Score int `json:"score"`
  4. }
  5. func main() {
  6. stu1 := student{
  7. Name: "小王子",
  8. Score: 90,
  9. }
  10. t := reflect.TypeOf(stu1)
  11. fmt.Println(t.Name(), t.Kind()) // student struct
  12. // 通过for循环遍历结构体的所有字段信息
  13. for i := 0; i < t.NumField(); i++ {
  14. field := t.Field(i)
  15. fmt.Printf("name:%s index:%d type:%v json tag:%v\n", field.Name, field.Index, field.Type, field.Tag.Get("json"))
  16. }
  17. // 通过字段名获取指定结构体字段信息
  18. if scoreField, ok := t.FieldByName("Score"); ok {
  19. fmt.Printf("name:%s index:%d type:%v json tag:%v\n", scoreField.Name, scoreField.Index, scoreField.Type, scoreField.Tag.Get("json"))
  20. }
  21. }

接下来编写一个函数printMethod(s interface{})来遍历打印s包含的方法。

  1. // 给student添加两个方法 Study和Sleep(注意首字母大写)
  2. func (s student) Study() string {
  3. msg := "好好学习,天天向上。"
  4. fmt.Println(msg)
  5. return msg
  6. }
  7. func (s student) Sleep() string {
  8. msg := "好好睡觉,快快长大。"
  9. fmt.Println(msg)
  10. return msg
  11. }
  12. func printMethod(x interface{}) {
  13. t := reflect.TypeOf(x)
  14. v := reflect.ValueOf(x)
  15. fmt.Println(t.NumMethod())
  16. for i := 0; i < v.NumMethod(); i++ {
  17. methodType := v.Method(i).Type()
  18. fmt.Printf("method name:%s\n", t.Method(i).Name)
  19. fmt.Printf("method:%s\n", methodType)
  20. // 通过反射调用方法传递的参数必须是 []reflect.Value 类型
  21. var args = []reflect.Value{}
  22. v.Method(i).Call(args)
  23. }
  24. }

call方法

结构体中的方法的排序是按照ASCII码来进行排序的。
调用call方法首先得获取到结构体得方法,获取结构体得方法通过tValue.Method(Num)
其中:

  • tValue是一个type.Value类型得变量
  • Method是调用得方法名
  • Num是结构体得方法排序后的索引值

例子:

  1. package main
  2. import (
  3. "fmt"
  4. "reflect"
  5. )
  6. // Person 定义一个struct
  7. type Person struct {
  8. Name string
  9. Age int
  10. Address string
  11. }
  12. // Set 给Person定义Set方法
  13. func (p *Person) Set(name, address string, age int) {
  14. p.Name = name
  15. p.Age = age
  16. p.Address = address
  17. }
  18. // Get 给Person定义Get方法
  19. func (p *Person) Get() {
  20. fmt.Println(p)
  21. }
  22. // Sum ...
  23. func (p *Person) Sum(a, b int) {
  24. fmt.Println(a + b)
  25. }
  26. // 定义一个函数来通过放射操作结构体
  27. func reflectTest(r interface{}) {
  28. // 通过反射获取type.Type
  29. // tType:=reflect.TypeOf(r)
  30. // 通过反射获取type.Value
  31. tValue := reflect.ValueOf(r)
  32. // fmt.Println(tValue.Elem().Kind())
  33. // 判断tValue的类型是否是reflect.Struct
  34. if tValue.Elem().Kind() != reflect.Struct {
  35. fmt.Println("the type of r must be struct")
  36. return
  37. }
  38. // 获取结构体总共有几个方法
  39. fNum := tValue.NumMethod()
  40. fmt.Println(fNum)
  41. // 调用方法
  42. tValue.Method(0).Call(nil)
  43. }
  44. func main() {
  45. // 实例化结构体
  46. p1 := &Person{
  47. Name: "Tom",
  48. Age: 30,
  49. Address: "Chongqing",
  50. }
  51. reflectTest(p1)
  52. }

最后输出的结果为:

  1. &{Tom 30 Chongqing}

我们上面定义的方法的顺序是Set()Get()Sum()。但是我们通过反射获取第0方法是Get()。就像我们上面说的,获取的方法是通过ASSII码排序后的方法。上面三个方法通过ASSII码排序后为Get()、Set()、Sum()。
如果方法是有参数的,传递的参数必须是[]reflect.Value类型。
如下:

  1. // 定义切片类型的value作为参数
  2. var param []reflect.Value
  3. param = append(param, reflect.ValueOf(10))
  4. param = append(param, reflect.ValueOf(20))
  5. tValue.Method(2).Call(param)