Go Web编程实战----反射
Posted 李元静
tags:
篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了Go Web编程实战----反射相关的知识,希望对你有一定的参考价值。
目录
反射
与其他语言一样,Go语言的反射同样是指,计算机程序在运行时,可以访问、检测和修改它本身状态或行为的一种能力。
其在reflect包里,定义了一个接口和一个结构体,即reflect.Type接口与reflect.Value结构体,它们提供了很多函数来获取存储在接口里的类型信息。
- reflect.Type接口:主要提供关于类型相关的信息
- reflect.Value结构体:主要提供关于值相关的信息,可以获取甚至改变类型的值。
reflect包中提供了两个基础的关于反射的函数,用来获取上述接口与结构体:
func TypeOf(i interface()) Type
func ValueOf(I interface()) Value
其中:
- TypeOf()函数:用来提取一个接口中值的类型信息。由于它的输入参数是一个空的interface,所以在调用此函数时,实参会先被转换为interface类型。这样,实参的类型信息、方法集、值信息都存储到interface变量里了。
- ValueOf()函数:返回一个结构体变量,包含类型信息及实际值。
详细原理图如下:
反射的3大原则
在Go语言中,反射有3大原则:
- 反射可以将”接口类型变量“转换为”反射类型对象“
- 反射可以将”反射类型对象“转换为”接口类型变量“
- 如果要修改”反射类型对象“,则其值必须是“可写的”
“接口类型变量”转换为“反射类型对象”
反射是一种检查存储在接口变量中的类型与值对的机制。reflect包中的两个类型:Type和Value。这2种类型给了我们访问一个接口变量种所包含的内容的途径。
另外,2个简单的函数reflect.TypeOf()和reflect.ValueOf()可以检索一个接口值的reflect.Type与reflect.Value部分。示例如下:
import (
"fmt"
"reflect"
)
func main()
var x float64 = -3.151592653
fmt.Println("Type:", reflect.TypeOf(x))
v := reflect.ValueOf(x)
fmt.Println("Value:", v)
fmt.Println("Type:", v.Type())
fmt.Println("is float64:", v.Kind() == reflect.Float64)
fmt.Println("Value:", v.Float())
运行之后,我们会得到如下结果。
上面代码,我们先是定义了一个float64的变量,然后将其赋值给reflect.TypeOf(x)函数。在我们调用该函数时,x会被保存到空接口中,然后这个空接口作为参数传递,reflect.TypeOf会将空接口拆包恢复出类型信息。
当然,reflect.ValueOf(x)同样也可以将值恢复出来,而且reflect.ValueOf获取的变量类型还可以使用方法进行操作。比如获取类型,值,以及对比类型。
“反射类型对象”转换为“接口类型变量”
这个与上面的正好相反,和物理学类似,有物质就有反物质。
在Go语言中,反射也能创造自己反面类型的对象。根据reflect.Value类型的变量,可以使用interface()方法恢复其接口类型的值。而且该方法,会把type和value信息打包并填充到一个接口变量中,然后返回。定义如下:
//定义
func (v Value) Interface() interface
//例子
func main()
var name interface="liyuanjing"
x :=reflect.TypeOf(name)
y :=reflect.ValueOf(name)
//从接口变量到反射对象
fmt.Printf("从接口变量到反射对象:Type对象的类型为%T \\n",x)
fmt.Printf("从接口变量到反射对象:Value对象的类型为%T \\n",y)
//从反射对象到接口变量
z :=y.Interface()
fmt.Printf("从反射对象到接口变量:新对象的类型为%T 值为%v \\n",z,z)
运行之后,控制台输出效果如下:
“反射类型对象”修改(值必“可写的”)
在使用reflect.TypeOf()函数和reflect.ValueOf()函数时,如果传递的不是接口变量的指针,则反射世界里的变量始终只是真实世界里的一个复制,对该反射对象进行修改,并不能反映到真实世界里。
需要注意的是:
- 不是接收变量指针创建的反射对象,是不具备“可写性”的
- 是否具备“可写性”,可使用CanSet()方法来判断
- 对不具备“可写性”的对象进行修改,是没有意义的,也认为是不合法的,因此会报错。
如果需要让反射具备可写性,需要这样:
- 创建反射对象时,传入变量的是指针
- 使用Elem()方法,返回指针指向的数据。
判断可写性示例:
func main()
var name string = "liyuanjing"
x := reflect.ValueOf(&name)
fmt.Println("x的可写性为:", x.CanSet())
y := x.Elem()
fmt.Println("y的可写性为:", y.CanSet())
运行之后,你会发现x是不可写的,y因为使用了Elem()方法,是可写的。
知道了如何使用反射世界里的对象具有可写性后,接下来是时候了解一下,如何对修改更新对象了。
在反射的Value对象中,有多个以单词Set开头的方法用于重新设置对应类型的值。比如:
func (v Value) SetBool(x bool)
func (v Value) SetBytes(x []byte)
func (v Value) setRunes(x []rune)
func (v Value) SetComplex(x complex128)
func (v Value) SetFloat(x float64)
func (v Value) SetInt(x int64)
这些方法全部都是修改值的入口,比如,通过反射对象SetInt()方法进行更新值的示例如下:
func main()
var num int = 30
fmt.Println("原始值为:", num)
x := reflect.ValueOf(&num)
y := x.Elem()
y.SetInt(500)
fmt.Println("通过反射对象进行更新后,num的真实值为:", num)
运行之后,效果如下:
以上是关于Go Web编程实战----反射的主要内容,如果未能解决你的问题,请参考以下文章