Go语言 详解reflect

原创文章,转载请注明出处:服务器非业余研究-sunface

首先,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 TypeOf和ValueOf是获取Type和Value的方法

2 ValueOf返回的<float64 Value>是为了说明这里的value是float64

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

Value的方法

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

下面看这么个例子:

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 c string = "yejianfeng"
    p := reflect.ValueOf(&c)
    fmt.Println(p.CanSet())   //false
    fmt.Println(p.Elem().CanSet())  //true
    p.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包是开发过程中几乎必备的包之一。能合理和熟练使用它对开发有很大的帮助。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值