反射 go

reflect.Valueof 到底是个什么?

 

 

 

反射值对象(reflect.Value)提供一系列方法进行零值和空判定,如下表所示。

反射值对象的零值和有效性判断方法
方 法说 明
IsNil() bool 返回值是否为 nil。如果值类型不是通道(channel)、函数、接口、map、指针或 切片时发生 panic,类似于语言层的v== nil操作
IsValid() bool 判断值是否有效。 当值本身非法时,返回 false,例如 reflect Value不包含任何值,值为 nil 等。


下面的例子将会对各种方式的空指针进行 IsNil() 和 IsValid() 的返回值判定检测。同时对结构体成员及方法查找 map 键值对的返回值进行 IsValid() 判定,参考下面的代码。

 

代码一:
反射值对象的零值和有效性判断:

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

代码输出如下:

var a *int: true
nil: false
(*int)(nil): false
不存在的结构体成员: false
不存在的结构体方法: false
不存在的键: false

代码说明如下:

  • 第 11 行,声明一个 *int 类型的指针,初始值为 nil。
  • 第 12 行,将变量 a 包装为 reflect.Value 并且判断是否为空,此时变量 a 为空指针,因此返回 true。
  • 第 15 行,对 nil 进行 IsValid() 判定(有效性判定),返回 false。
  • 第 18 行,(*int)(nil) 的含义是将 nil 转换为 *int,也就是*int 类型的空指针。此行将 nil 转换为 *int 类型,并取指针指向元素。由于 nil 不指向任何元素,*int 类型的 nil 也不能指向任何元素,值不是有效的。因此这个反射值使用 Isvalid() 判断时返回 false。
  • 第 21 行,实例化一个结构体。
  • 第 24 行,通过 FieldByName 查找 s 结构体中一个空字符串的成员,如成员不存在,IsValid() 返回 false。
  • 第 27 行,通过 MethodByName 查找 s 结构体中一个空字符串的方法,如方法不存在,IsValid() 返回 false。
  • 第 30 行,实例化一个 map,这种写法与 make 方式创建的 map 等效。
  • 第 33 行,MapIndex() 方法能根据给定的 reflect.Value 类型的值查找 map,并且返回查找到的结果。


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

 

 

 

 

 

 

 

 

 

 

代码二:

 

    a := map[string]int{}
    a["aa"] = 2
    b := reflect.ValueOf(a).MapIndex(reflect.ValueOf("aa"))
    c := b.Kind()
    fmt.Println(c == reflect.Int)//打印true
posted @ 2019-12-04 01:42  扫驴  阅读(301)  评论(0编辑  收藏  举报