Go语言_反射篇

简介:

这里的GO使用的版本是1.2

Go语言的基本语法的使用已经在前几篇陆陆续续学完了,下面可能想写一些Go的标准库的使用了。

先是reflect库。

reflect库的godoc在http://golang.org/pkg/reflect/

Type和Value

首先,reflect包有两个数据类型我们必须知道,一个是Type,一个是Value。

Type就是定义的类型的一个数据类型,Value是值的类型

具体的Type和Value里面包含的方法就要看文档了:

http://golang.org/pkg/reflect/

 

这里我写了个程序来理解Type和Value:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
package main
 
import(
     "fmt"
     "reflect"
)
 
type MyStruct  struct {
     name  string
}
 
func ( this  *MyStruct)GetName()  string  {
     return  this .name
}
 
func main() {
     s :=  "this is string"
     fmt.Println(reflect.TypeOf(s))
     fmt.Println( "-------------------" )
     
     fmt.Println(reflect.ValueOf(s))
     var  x float64 = 3.4
     fmt.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
2
3
4
5
6
7
8
string
-------------------
this  is  string
<float64 value>
-------------------
1
-------------------
yejianfeng

 

补充,在Go version 1.5中会返回

1
2
3
4
5
6
7
8
string
-------------------
this  is  string
3.4
-------------------
1
-------------------
yejianfeng

  

这个程序看到几点:

1 TypeOf和ValueOf是获取Type和Value的方法

2 第三个b的定义实现了php中的string->method的方法,为什么返回的是reflect.Value[]数组呢?当然是因为Go的函数可以返回多个值的原因了。

Value的方法和属性

好了,我们看到Value的Type定义了这么多Set方法:

clip_image002

下面看这么个例子:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
package main
 
import(
     "fmt"
     "reflect"
)
 
type MyStruct  struct {
     name  string
}
 
func ( this  *MyStruct)GetName()  string  {
     return  this .name
}
 
func main() {
     fmt.Println( "--------------" )
     var  a MyStruct
     b :=  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( "--------------" )
     var  string  "yejianfeng"
     p := reflect.ValueOf(&c)
     fmt.Println(p.CanSet())    //false
     fmt.Println(p.Elem().CanSet())   //true
     p.Elem().SetString( "newName" )
     fmt.Println(c)
}

返回:

clip_image003

 

这段代码能有一些事情值得琢磨:

1 为什么a和b的ValueOf返回的是不一样的?

a是一个结构,b是一个指针。好吧,在Go中,指针的定义和C中是一样的。

2 reflect.ValueOf(a).FieldByName("name")

这是一个绕路的写法,其实和a.name是一样的意思,主要是要说明一下Value.FieldByName的用法

3 val := reflect.ValueOf(b).FieldByName("name") 是有error的,为什么?

b是一个指针,指针的ValueOf返回的是指针的Type,它是没有Field的,所以也就不能使用FieldByName

4 fmt.Println(reflect.ValueOf(a).FieldByName("name").CanSet())为什么是false?

看文档中的解释:

clip_image004

好吧,什么是addressable,and was not obtained by the use of unexported struct fields?

CanSet当Value是可寻址的时候,返回true,否则返回false

看到第二个c和p的例子,我们可以这么理解:

当前面的CanSet是一个指针的时候(p)它是不可寻址的,但是当是p.Elem()(实际上就是*p),它就是可以寻址的

这个确实有点绕。

 

总而言之,reflect包是开发过程中几乎必备的包之一。能合理和熟练使用它对开发有很大的帮助。

 

20160829 补充:

Go 1.5的reflect Type方法可以看下面这个例子:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
package main
 
import (
     "fmt"
     "reflect"
)
 
type MyStruct  struct  {
     name  string
}
 
func ( this  *MyStruct) GetName()  string  {
     return  this .name
}
 
type IStruct  interface  {
     GetName()  string
}
 
func main() {
     // TypeOf
     s :=  "this is string"
     fmt.Println(reflect.TypeOf(s))  // output: "string"
 
     // object TypeOf
     a :=  new (MyStruct)
     a.name =  "yejianfeng"
     typ := reflect.TypeOf(a)
     fmt.Println(typ)         // output: "*main.MyStruct"
     fmt.Println(typ.Elem())  // output: "main.MyStruct"
 
     // reflect.Type Base struct
     fmt.Println(typ.NumMethod())                    // 1
     fmt.Println(typ.Method(0))                      // {GetName  func(*main.MyStruct) string <func(*main.MyStruct) string Value> 0}
     fmt.Println(typ.Name())                         // ""
     fmt.Println(typ.PkgPath())                      // ""
     fmt.Println(typ.Size())                         // 8
     fmt.Println(typ.String())                       // *main.MyStruct
     fmt.Println(typ.Elem().String())                // main.MyStruct
     fmt.Println(typ.Elem().FieldByIndex([] int {0}))  // {name main string  0 [0] false}
     fmt.Println(typ.Elem().FieldByName( "name" ))     // {name main string  0 [0] false} true
 
     fmt.Println(typ.Kind() == reflect.Ptr)                               // true
     fmt.Println(typ.Elem().Kind() == reflect.Struct)                     // true
     fmt.Println(typ.Implements(reflect.TypeOf((*IStruct)(nil)).Elem()))  // true
 
     fmt.Println(reflect.TypeOf(12.12).Bits())  // 64, 因为是float64
 
     cha := make(chan  int )
     fmt.Println(reflect.TypeOf(cha).ChanDir())  // chan
 
     var  fun func(x  int , y ...float64)  string
     var  fun2 func(x  int , y float64)  string
     fmt.Println(reflect.TypeOf(fun).IsVariadic())   // true
     fmt.Println(reflect.TypeOf(fun2).IsVariadic())  // false
     fmt.Println(reflect.TypeOf(fun).In(0))          // int
     fmt.Println(reflect.TypeOf(fun).In(1))          // []float64
     fmt.Println(reflect.TypeOf(fun).NumIn())        // 2
     fmt.Println(reflect.TypeOf(fun).NumOut())       // 1
     fmt.Println(reflect.TypeOf(fun).Out(0))         // string
 
     mp := make(map[ string ] int )
     mp[ "test1" ] = 1
     fmt.Println(reflect.TypeOf(mp).Key())  //string
 
     arr := [1] string { "test" }
     fmt.Println(reflect.TypeOf(arr).Len())  // 1
 
     fmt.Println(typ.Elem().NumField())  // 1
 
     // MethodByName, Call
     b := reflect.ValueOf(a).MethodByName( "GetName" ).Call([]reflect.Value{})
     fmt.Println(b[0])  // output: "yejianfeng"
}

 





本文转自轩脉刃博客园博客,原文链接:http://www.cnblogs.com/yjf512/archive/2012/06/10/2544391.html,如需转载请自行联系原作者

相关文章
|
16天前
|
存储 Go 索引
go语言中数组和切片
go语言中数组和切片
26 7
|
15天前
|
Go 开发工具
百炼-千问模型通过openai接口构建assistant 等 go语言
由于阿里百炼平台通义千问大模型没有完善的go语言兼容openapi示例,并且官方答复assistant是不兼容openapi sdk的。 实际使用中发现是能够支持的,所以自己写了一个demo test示例,给大家做一个参考。
|
16天前
|
程序员 Go
go语言中结构体(Struct)
go语言中结构体(Struct)
92 71
|
15天前
|
存储 Go 索引
go语言中的数组(Array)
go语言中的数组(Array)
100 67
|
18天前
|
Go 索引
go语言for遍历数组或切片
go语言for遍历数组或切片
88 62
|
19天前
|
并行计算 安全 Go
Go语言中的并发编程:掌握goroutines和channels####
本文深入探讨了Go语言中并发编程的核心概念——goroutine和channel。不同于传统的线程模型,Go通过轻量级的goroutine和通信机制channel,实现了高效的并发处理。我们将从基础概念开始,逐步深入到实际应用案例,揭示如何在Go语言中优雅地实现并发控制和数据同步。 ####
|
16天前
|
存储 Go
go语言中映射
go语言中映射
32 11
|
18天前
|
Go
go语言for遍历映射(map)
go语言for遍历映射(map)
29 12
|
17天前
|
Go 索引
go语言使用索引遍历
go语言使用索引遍历
26 9
|
21天前
|
安全 Serverless Go
Go语言中的并发编程:深入理解与实践####
本文旨在为读者提供一个关于Go语言并发编程的全面指南。我们将从并发的基本概念讲起,逐步深入到Go语言特有的goroutine和channel机制,探讨它们如何简化多线程编程的复杂性。通过实例演示和代码分析,本文将揭示Go语言在处理并发任务时的优势,以及如何在实际项目中高效利用这些特性来提升性能和响应速度。无论你是Go语言的初学者还是有一定经验的开发者,本文都将为你提供有价值的见解和实用的技巧。 ####