在Go语言中反射不仅可以获取值的类型和种类,还可以获取值和更改值,使用reflect.ValueOf()获取和设置变量的值。
使用反射值包装任意值
Go语言通过reflect.ValueOf()获取的是值的反射值对象,书写格式如下
value := reflect.ValueOf(rawValue)
reflect.ValueOf()返回的是reflect.Value()类型。reflect.Value()与原值间可以通过值包装和值获取相互转化。
从反射值对象获取被包装的值
从反射值对象(reflect.Value)中获取原值有几种方法如下:
方法名 说 明
Interface() interface{} 将值以interface{}类型返回,可以通过类型断言转换为指定类型
Int() int64 将值以int类型返回,所有有符号整数均可以以此方法返回
Uint() uint64 将值以uint类型返回,所有无符号整数均可以以此方法返回
Float() float64 将值以双精度类型返回,所有浮点数均可以以此方法返回
Bool() bool 将值以bool类型返回
Bytes() []bytes 将值以字节数组类型返回
String() string 将值以字符串类型返回
从反射值对象获取值的代码示例
package main
import (
"fmt"
"reflect"
)
func main() {
//声明一个整型变量a并赋值
var a int = 1024
//获取a的反射值对象
valueOf := reflect.ValueOf(a)
//获取interface的类型值,通过类型断言进行转换
var getA int := valueOf.interface().(int)
//通过强制类型转换为int类型
var getB := int(valueOf.Int())
//输出两种值
fmt.Println(getA,getB)
使用反射访问结构体的成员字段值
反射值对象提供对结构体访问的方法,这些方法可以完成对结构体任意值的访问。
方 法 备注
Field(i int) Value 根据索引,返回索引对应的结构体成员字段的反射值对象。当值不是结构体或者索引超界时发生宕机
NumField() int 返回结构体成员字段数量。当值不是结构体或者索引超届时发生宕机
FieldByName(name string) Value 根据给定字符串返回字符串对应的结构体字段。没有找到时返回零值当值不是结构体或索引超界时宕机
FieldByNameFunc(match func(string) bool)
Value 根据匹配函数匹配需要的字段。找到时返回零值,当值不是结构体或索引超界时宕机
通过反射访问结构体成员值代码示例
package main
import (
"fmt"
"reflect"
)
func main() {
//定义结构体
type wy struct {
a int
b string
//嵌入字段
float32
bool
next *wy
}
//获取结构体反射值对象
d := reflect.ValueOf(wy{next: &wy{}})
//获取字段数量
fmt.Println("NumField:",d.NumField())
//获取索引为2的字段
floatField := d.Field(2)
fmt.Println("Field:",floatField.Type())
//根据名字查找字段
fmt.Println("FieldByName(\"b\").Type",d.FieldByName("b").Type())
//根据索引值找字段中next的int字段值
fmt.Println("FieldByIndex([]int{4,0}).Type()", d.FieldByIndex([]int{4,0}).Type())
反射值对象中使用isNil() bool 内置函数判断是否为空,用isValid()函数判断值是否有效。
使用反射值对象修改变量的值
反射值对象判定及获取元素的方法
方法 备注
Elem() Value 取值指向的元素值,类似于语言层 “*”操作。当值类型不是指针或者接口时发生宕机,空指针时返回nil的Value
Addr() Value 对可寻址的值返回其地址,类似于语言层“&”操作。当值不可寻址时发生宕机
CanAddr() bool 表示值是否可寻址
CanSet() bool 返回值能否被修改。要求值可寻址且是导出的字段
使用反射更改值时,首先要确定被更改的值被寻址。
使用反射更改值的代码示例
package main
import(
"fmt"
"reflect"
)
func main() {
//声明一个整型变量a并赋值
var a int = 2
//获取a的反射值对象
valueOfA := reflect.ValueOf(&a)
//取出a元素的地址
valueOfA = ValueOfA.Elem()
//修改a的值为1
valueOfA.SetInt(1)
//打印输出
fmt.Println(valueOfA.Int())
当reflect.Value不可寻址时,使用Addr()方法也是不可寻址的,同时发生宕机。
在结构体中成员值如果想用反射更改值需要先导出。
代码示例如下:
package main
import(
"fmt"
"reflect"
)
func main() {
type dog struct {
LegCount int
}
//获取dog实例地址的反射值对象
valueOfDog := reflect.ValueOf(&dog{})
//获取dog实例地址元素
valueOfDog = valueOfDog.Elem()
//获取LegCount字段(结构体成员导出就是将成员名首字母大写)
vLegCount := valueOfDog.FieldByName("LegCount")
//尝试设置LegCount的值
vLegCount.SetInt(4)
//d打印输出
fmt.Println(vLegCount.Int())