GO语言--反射

定义

计算机在运行时(Run Time)可以访问,检测和修改它本身状态或行为的一种能力。简单来说,反射就是程序在运行时能够观察并修改自己的行为。

这里的运行时是一个名词,指的是一个程序正在在运行或在被执行中的状态。

常用场景

1.在定义函数时,函数参数没有设置特定的数据类型(比如将参数设置为空接口)。如果需要对参数的数据类型或参数值进行判断,可以使用反射实现。

2.在调用函数时,根据if条件调用对应函数,可以对函数或参数进行反射,在运行期间能够动态执行函数调用。

第一定律:接口变量转反射变量

我们之前提到过空接口的定义及使用--空接口的数据类型是动态可变的。因此空接口和反射机制之间能够灵活转换。

反射机制需要导入"reflect"包。reflect包的两种基本类型为:Type和Value,分别对应两个方法reflect.TypeOf()和reflect.ValueOf(),用来读取接口卡变量的数据和类型。例子如下

package main

import (
	"fmt"
	"reflect"
)

func main() {
	test := []interface{}{1, "ishmael", 2.3}
	it := reflect.TypeOf(test)
	fmt.Printf("%T\n", it)
	fmt.Printf("%v\n", it)
	iv := reflect.ValueOf(test)
	fmt.Printf("%T\n", iv)
	fmt.Printf("%v\n", iv)
}

运行结果为

 *reflect.rtype  //反射变量的类型
[]interface {} //转换为反射变量前的变量类型
reflect.Value  //反射变量的类型
[1 ishmael 2.3] //变量的数值

 当然,虽然说是接口变量转反射变量。但由于空接口的数据类型灵活可变,实际上int,string也能直接转为反射变量。例子如下

package main

import (
	"fmt"
	"reflect"
)

func main() {

	it := reflect.TypeOf("ishmael")
	fmt.Printf("%T\n", it)
	fmt.Printf("%v\n", it)
	iv := reflect.ValueOf("ishmael")
	fmt.Printf("%T\n", iv)
	fmt.Printf("%v\n", iv)
}

运行结果为

*reflect.rtype
string       
reflect.Value
ishmael

扫描二维码关注公众号,回复: 15748891 查看本文章

第二定律:反射变量转接口变量

我们上面提到过,接口变量与反射变量是能相互转换的。只需要调用结构体方法interface()即可。比较简单,例子如下

package main

import (
	"fmt"
	"reflect"
)

func main() {

	name := "ishmael"
	iv := reflect.ValueOf(name)
	fmt.Printf("Type is %T\nValue is %v\n", iv, iv)
	i := iv.Interface()
	fmt.Printf("Type is %T\nValue is %v\n", i, i)
}

运行结果

Type is reflect.Value
Value is ishmael
Type is string  
Value is ishmael

第三定律:修改反射变量的值 

在开头我们提到过,反射机制主要是用于程序在运行时,我们想去修改某个变量以满足对应的功能。所以我们才会将变量转换为反射变量。

但是,当我们将某个变量转换为反射变量时,反射变量只拷贝了变量的值。相当于我们新建了一个变量,我们后续的操作都是对这个新建的变量执行的。所以即使我们修改了反射变量,原变量的值也不会发生任何改变。这明显与我们的反射的定义不符。

计算机在运行时可以访问,检测和修改它本身状态或行为的一种能力

那我们怎么解决这个问题呢? 为了解决这种情况,GO语言为反射变量设置了可写状态不可写状态,从而有了CanSet()方法以判断当前变量能否被修改。

如果我们直接传入一个变量,那它是不可写的。那怎么才能通过修改反射变量的值同时修改原变量的值呢?

我们之前学过指针。那么我们可以在使用reflect.ValueOf()时传入指针。通过指针,我们可以修改内存地址对应的数值,使反射变量和整型变量的数值达成一致。

需要注意的是,由于我们传入的是地址,所以需要Elem()方法来获取该内存地址上的值,对这个值调用CanSet()方法,判断其是否可写。同时修改反射变量时不能修改为不同的数据类型,如string类型的反射变量不能被修改为int类型,否则会报错。例子如下

package main

import (
	"fmt"
	"reflect"
)

func main() {

	name := "ishmael"
	it := reflect.ValueOf(&name)
	fmt.Printf("Value is %v, Type is %T\n", it, it)
	iv := it.Elem() //通过Elem()方法,获取该内存地址上的值
	fmt.Printf("Value is %v, Type is %T\n", iv, iv)
	if iv.CanSet() {
		iv.SetString("faust")
		fmt.Printf("Value is %v, Type is %T\n", iv, iv)
		iv.SetInt(10)
		fmt.Printf("Value is %v, Type is %T\n", iv, iv) //修改为不同类型会报错
	}

}

运行结果

Value is 0xc000088020, Type is reflect.Value
Value is ishmael, Type is reflect.Value                                  
Value is faust, Type is reflect.Value                                    
panic: reflect: call of reflect.Value.SetInt on string Value 

猜你喜欢

转载自blog.csdn.net/white_night_SZTU/article/details/129642863
今日推荐