反射
反射可以支持 不知道对象的类信息下访问类的方法和属性
再举个常见例子,我和你编写同一个动物园软件,分开写不同模块,你负责写饲养员部分,我负责写动物模块,我会传给你一个动物对象,你负责调教好它,我们事先完全解耦,写作过程中也完全没有沟通,我传给你一个黑盒子对象你只能自己用反射摸索,比如你查询我传给你的对象有哪些方法:你发现有一个“飞”方法,那你可能在养一只鸟,这时你可以调用这个方法让它飞一飞,然后你又用反射得到另一个对象有“游”方法,这时你就可以把它放到水里面养一养了。
反射机制由浅入深包含
(1)运行时获取未知对象的方法、属性信息(比如方法的参数、属性的名字类型等)
(2)运行时调用未知对象的方法、访问它的属性
(3)运行时实例化一个未知类型(也就是没有代码定义过类)的对象
go语言是支持反射的,关于反射的方法都在reflect这个包里面,主要用的是TypeOf和ValueOf这两个,分别返回Type类型和Value类型
其基本的用法如下面所示
package main
import (
"fmt"
"reflect"
)
type Student struct{
Name string
Age int
}
func (s Student) Set(name string, age int) {
s.Name = name
s.Age = age
}
//函数必须大写
func (s Student) Print(){
fmt.Println("print --------start")
fmt.Println(s.Name)
fmt.Println(s.Age)
fmt.Println("print -------- end")
}
func test(a interface{}){
v := reflect.TypeOf(a)
fmt.Println(v)
b := reflect.ValueOf(a)
//c := b.String()
c := b.Kind()
fmt.Println(c)
fmt.Println(b)
}
//反射改变基本的数据类型
func Test(a interface{}){
v := reflect.TypeOf(a)
fmt.Println(v)
b := reflect.ValueOf(a)
fmt.Println(b)
b.Elem().SetInt(11)
//不能用*b来获取对象 要用Elem()函数来获取指针指向的对象
fmt.Println(b.Elem().Int())
}
//反射操作结构体类型
func TestStruct(a interface{}){
//判断是不是结构体
p := reflect.ValueOf(a)
val := p.Elem()
kd := val.Kind()
if kd != reflect.Struct{
fmt.Println("not struct failed!!")
return
}
//对结构体进行操作
num := val.NumField()
for i:=0; i<num; i++{
fmt.Printf("struct index :%d value:%v kind:%s \n",i,val.Field(i),val.Field(i).Kind())
}
fmt.Println(num)/*获取结构体字段的个数*/
//修改结构体里面的值
val.Field(0).SetString("bobo")
val.Field(1).SetInt(1234)
for i:=0; i<num; i++{
fmt.Printf("struct index :%d value:%v\n",i,val.Field(i))
}
fmt.Println(val.NumMethod())/*获取结构体当中方法的个数*/
methodnum := val.NumMethod()
for i:=0; i<methodnum; i++{
fmt.Printf("method index :%d value:%v\n",i,val.Method(i))
}
//调用结构体的方法
val.Method(0).Call(nil)
}
func main(){
var stu Student = Student{
Name:"echoqian",
Age:21,
}
test(stu)
var a int = 21
Test(&a)
TestStruct(&stu)
}
代码运行结果
main.Student
struct
{echoqian 21}
*int
0xc0000120b0
11
struct index :0 value:echoqian kind:string
struct index :1 value:21 kind:int
2
struct index :0 value:bobo
struct index :1 value:1234
2
method index :0 value:0x49d070
method index :1 value:0x49d070
print --------start
bobo
1234
print -------- end