go语言的基本语法的使用已经在前几篇陆陆续续学完了,下面可能想写一些go的标准库的使用了。
先是reflect库。
首先,reflect包有两个数据类型我们必须知道,一个是type,一个是value。
type就是定义的类型的一个数据类型,value是值的类型
具体的type和value里面包含的方法就要看文档了:
<a href="http://golang.org/pkg/reflect/">http://golang.org/pkg/reflect/</a>
这里我写了个程序来理解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
<code>package main</code>
<code>import(</code>
<code> </code><code>"fmt"</code>
<code> </code><code>"reflect"</code>
<code>)</code>
<code>type mystruct </code><code>struct</code><code>{</code>
<code> </code><code>name </code><code>string</code>
<code>}</code>
<code>func (</code><code>this</code> <code>*mystruct)getname() </code><code>string</code> <code>{</code>
<code> </code><code>return</code> <code>this</code><code>.name</code>
<code>func main() {</code>
<code> </code><code>s := </code><code>"this is string"</code>
<code> </code><code>fmt.println(reflect.typeof(s))</code>
<code> </code><code>fmt.println(</code><code>"-------------------"</code><code>)</code>
<code> </code>
<code> </code><code>fmt.println(reflect.valueof(s))</code>
<code> </code><code>var</code> <code>x float64 = 3.4</code>
<code> </code><code>fmt.println(reflect.valueof(x))</code>
<code> </code><code>a := </code><code>new</code><code>(mystruct)</code>
<code> </code><code>a.name = </code><code>"yejianfeng"</code>
<code> </code><code>typ := reflect.typeof(a)</code>
<code> </code><code>fmt.println(typ.nummethod())</code>
<code> </code><code>b := reflect.valueof(a).methodbyname(</code><code>"getname"</code><code>).call([]reflect.value{})</code>
<code> </code><code>fmt.println(b[0])</code>
输出结果:

这个程序看到几点:
好了,我们看到value的type定义了这么多set方法:
下面看这么个例子:
37
38
39
40
41
42
<code> </code><code>fmt.println(</code><code>"--------------"</code><code>)</code>
<code> </code><code>var</code> <code>a mystruct</code>
<code> </code><code>b := </code><code>new</code><code>(mystruct)</code>
<code> </code><code>fmt.println(reflect.valueof(a))</code>
<code> </code><code>fmt.println(reflect.valueof(b))</code>
<code> </code><code>b.name = </code><code>"yejianfeng"</code>
<code> </code><code>val := reflect.valueof(a).fieldbyname(</code><code>"name"</code><code>)</code>
<code> </code><code>//painc: val := reflect.valueof(b).fieldbyname("name")</code>
<code> </code><code>fmt.println(val)</code>
<code> </code><code>fmt.println(reflect.valueof(a).fieldbyname(</code><code>"name"</code><code>).canset())</code>
<code> </code><code>fmt.println(reflect.valueof(&(a.name)).elem().canset())</code>
<code> </code><code>var</code> <code>c </code><code>string</code> <code>= </code><code>"yejianfeng"</code>
<code> </code><code>p := reflect.valueof(&c)</code>
<code> </code><code>fmt.println(p.canset()) </code><code>//false</code>
<code> </code><code>fmt.println(p.elem().canset()) </code><code>//true</code>
<code> </code><code>p.elem().setstring(</code><code>"newname"</code><code>)</code>
<code> </code><code>fmt.println(c)</code>
返回:
这段代码能有一些事情值得琢磨:
a是一个结构,b是一个指针。好吧,在go中,指针的定义和c中是一样的。
这是一个绕路的写法,其实和a.name是一样的意思,主要是要说明一下value.fieldbyname的用法
b是一个指针,指针的valueof返回的是指针的type,它是没有field的,所以也就不能使用fieldbyname
看文档中的解释:
好吧,什么是addressable,and was not obtained by the use of unexported struct fields?
canset当value是可寻址的时候,返回true,否则返回false
看到第二个c和p的例子,我们可以这么理解:
这个确实有点绕。
总而言之,reflect包是开发过程中几乎必备的包之一。能合理和熟练使用它对开发有很大的帮助。