go中的反射reflect提供了两种类型,其中type可以用来获取数据的类型,而value可以用来接收数据的值。
go中的所有函数调用都是值传递,所以当我们使用如下方法获取x的值时,(reflect.ValueOf()会将传入的参数转换为reflect.Valuel类型,再赋值给目标参数)
var x float64 = 3.4 v := reflect.ValueOf(x) fmt.Println(v.CanSet()) // false
会发现我们无法通过对v的操作来改变x的值,因为v只是x的一个值的副本而已,我们可以使用Canset()函数来判断是否能通过 v 来设置这里的 x 变量。
如果我们将x的地址传给v呢,能否通过副本来改变本体呢?
var x float64 = 3.4 v := reflect.ValueOf(&x) fmt.Println(v.CanSet()) // false
这里的 v 指向的是 x 的指针。所以 CanSet 方法判断的是 x 的指针是否可以设置。指针是肯定不能设置的,所以这里还是返回 false。
那么我们应该如何通过v来设置x呢,go提供了一个内置方法Elem()
var x float64 = 3.4 v := reflect.ValueOf(&x) fmt.Println(v.Elem().CanSet()) // true
但是这个 Elem() 使用的时候有个前提,这里的 value 必须是指针对象转换的 reflect.Value。(或者是接口对象转换的 reflect.Value)。这个前提不难理解吧,如果是一个 int 类型,它怎么可能有指向的元素呢?所以,使用 Elem 的时候要十分注意这点,因为如果不满足这个前提,Elem 是直接触发 panic 的。
var x float64 = 3.4 v := reflect.ValueOf(&x) if v.Elem().CanSet() { v.Elem().SetFloat(7.1) } fmt.Println(x) //7.1