Go语言 详解reflect
abv123456789 · · 7340 次点击 · · 开始浏览原创文章,转载请注明出处:服务器非业余研究-sunface
首先,reflect包有两个数据类型是最重要的,一个是Type,一个是Value。
Type就是定义的类型的一个数据类型,Value是值的类型
具体的Type和Value里面包含的方法就要看文档了:
http://golang.org/pkg/reflect/
这里有个方便同学们理解Type和Value的含义的具体示例:
123456789101112131415161718192021222324252627282930313233343536package mainimport("fmt""reflect")type MyStructstruct{namestring}func (this*MyStruct)GetName()string{returnthis.name}func main() {s :="this is string"fmt.Println(reflect.TypeOf(s))fmt.Println("-------------------")fmt.Println(reflect.ValueOf(s))varx float64 = 3.4fmt.Println(reflect.ValueOf(x))fmt.Println("-------------------")a :=new(MyStruct)a.name ="yejianfeng"typ := reflect.TypeOf(a)fmt.Println(typ.NumMethod())fmt.Println("-------------------")b := reflect.ValueOf(a).MethodByName("GetName").Call([]reflect.Value{})fmt.Println(b[0])}输出结果:
这个程序能看到以下几点:
1 TypeOf和ValueOf是获取Type和Value的方法
2 ValueOf返回的<float64 Value>是为了说明这里的value是float64
3 第三个b的定义实现了php中的string->method的方法,为什么返回的是reflect.Value[]数组呢?当然是因为Go的函数可以返回多个值的原因了。
Value的方法
好了,我们看到Value的Type定义了这么多Set方法:
下面看这么个例子:
123456789101112131415161718192021222324252627282930313233343536373839404142package mainimport("fmt""reflect")type MyStructstruct{namestring}func (this*MyStruct)GetName()string{returnthis.name}func main() {fmt.Println("--------------")vara MyStructb :=new(MyStruct)fmt.Println(reflect.ValueOf(a))fmt.Println(reflect.ValueOf(b))fmt.Println("--------------")a.name ="yejianfeng"b.name ="yejianfeng"val := reflect.ValueOf(a).FieldByName("name")//painc: val := reflect.ValueOf(b).FieldByName("name")fmt.Println(val)fmt.Println("--------------")fmt.Println(reflect.ValueOf(a).FieldByName("name").CanSet())fmt.Println(reflect.ValueOf(&(a.name)).Elem().CanSet())fmt.Println("--------------")varcstring="yejianfeng"p := reflect.ValueOf(&c)fmt.Println(p.CanSet())//falsefmt.Println(p.Elem().CanSet())//truep.Elem().SetString("newName")fmt.Println(c)}返回:
这段代码有些地方还是值得琢磨的:
1 为什么a和b的ValueOf返回是不一样的?
a是一个结构,b是一个指针。
2 reflect.ValueOf(a).FieldByName("name")
这是种写法较为繁琐,其实和a.name是一样的意思,主要是要说明一下嵌套式的用法
3 val := reflect.ValueOf(b).FieldByName("name") 是有error的,为什么?
b是一个指针,指针的ValueOf返回的是指针的Type,它是没有Field的,所以也就不能使用FieldByName
4 fmt.Println(reflect.ValueOf(a).FieldByName("name").CanSet())为什么是false?
看文档中的解释:
当Value不能被寻址或者使用了未导出struct fields的时候,都会返回false
看到第二个c和p的例子,我们可以这么理解:
当前面的CanSet是一个指针的时候(p)它是不可寻址的,但是当是p.Elem()(实际上就是*p),它就是可以寻址的.
总而言之,reflect包是开发过程中几乎必备的包之一。能合理和熟练使用它对开发有很大的帮助。
有疑问加站长微信联系(非本文作者)
入群交流(和以上内容无关):加入Go大咖交流群,或添加微信:liuxiaoyan-s 备注:入群;或加QQ群:692541889
关注微信- 请尽量让自己的回复能够对别人有帮助
- 支持 Markdown 格式, **粗体**、~~删除线~~、
`单行代码` - 支持 @ 本站用户;支持表情(输入 : 提示),见 Emoji cheat sheet
- 图片支持拖拽、截图粘贴等方式上传
收入到我管理的专栏 新建专栏
原创文章,转载请注明出处:服务器非业余研究-sunface
首先,reflect包有两个数据类型是最重要的,一个是Type,一个是Value。
Type就是定义的类型的一个数据类型,Value是值的类型
具体的Type和Value里面包含的方法就要看文档了:
http://golang.org/pkg/reflect/
这里有个方便同学们理解Type和Value的含义的具体示例:
123456789101112131415161718192021222324252627282930313233343536package mainimport("fmt""reflect")type MyStructstruct{namestring}func (this*MyStruct)GetName()string{returnthis.name}func main() {s :="this is string"fmt.Println(reflect.TypeOf(s))fmt.Println("-------------------")fmt.Println(reflect.ValueOf(s))varx float64 = 3.4fmt.Println(reflect.ValueOf(x))fmt.Println("-------------------")a :=new(MyStruct)a.name ="yejianfeng"typ := reflect.TypeOf(a)fmt.Println(typ.NumMethod())fmt.Println("-------------------")b := reflect.ValueOf(a).MethodByName("GetName").Call([]reflect.Value{})fmt.Println(b[0])}输出结果:
这个程序能看到以下几点:
1 TypeOf和ValueOf是获取Type和Value的方法
2 ValueOf返回的<float64 Value>是为了说明这里的value是float64
3 第三个b的定义实现了php中的string->method的方法,为什么返回的是reflect.Value[]数组呢?当然是因为Go的函数可以返回多个值的原因了。
Value的方法
好了,我们看到Value的Type定义了这么多Set方法:
下面看这么个例子:
123456789101112131415161718192021222324252627282930313233343536373839404142package mainimport("fmt""reflect")type MyStructstruct{namestring}func (this*MyStruct)GetName()string{returnthis.name}func main() {fmt.Println("--------------")vara MyStructb :=new(MyStruct)fmt.Println(reflect.ValueOf(a))fmt.Println(reflect.ValueOf(b))fmt.Println("--------------")a.name ="yejianfeng"b.name ="yejianfeng"val := reflect.ValueOf(a).FieldByName("name")//painc: val := reflect.ValueOf(b).FieldByName("name")fmt.Println(val)fmt.Println("--------------")fmt.Println(reflect.ValueOf(a).FieldByName("name").CanSet())fmt.Println(reflect.ValueOf(&(a.name)).Elem().CanSet())fmt.Println("--------------")varcstring="yejianfeng"p := reflect.ValueOf(&c)fmt.Println(p.CanSet())//falsefmt.Println(p.Elem().CanSet())//truep.Elem().SetString("newName")fmt.Println(c)}返回:
这段代码有些地方还是值得琢磨的:
1 为什么a和b的ValueOf返回是不一样的?
a是一个结构,b是一个指针。
2 reflect.ValueOf(a).FieldByName("name")
这是种写法较为繁琐,其实和a.name是一样的意思,主要是要说明一下嵌套式的用法
3 val := reflect.ValueOf(b).FieldByName("name") 是有error的,为什么?
b是一个指针,指针的ValueOf返回的是指针的Type,它是没有Field的,所以也就不能使用FieldByName
4 fmt.Println(reflect.ValueOf(a).FieldByName("name").CanSet())为什么是false?
看文档中的解释:
当Value不能被寻址或者使用了未导出struct fields的时候,都会返回false
看到第二个c和p的例子,我们可以这么理解:
当前面的CanSet是一个指针的时候(p)它是不可寻址的,但是当是p.Elem()(实际上就是*p),它就是可以寻址的.
总而言之,reflect包是开发过程中几乎必备的包之一。能合理和熟练使用它对开发有很大的帮助。