|
1 | | -#7.2 JSON处理 |
2 | | -JSON(Javascript Object Notation)是一种轻量级的数据交换语言,以文字为基础,具有自我描述性且易于让人阅读。尽管JSON是在Javascript的一个子集,但JSON是独立于语言的文本格式,并且采用了类似于C语言家族的一些习惯。JSON与XML最大的不同在于XML是一个完整的标记语言,而JSON不是。JSON由于比XML更小、更快,更易解析,以及浏览器的内建快速解析支持,使得其更适用于网络数据传输领域。目前我们看到很多的开放平台,基本上所有的接口都是采用了JSON作为他们的数据交互。那么JSON在Web开发中如此重要,Go语言对于JSON支持的怎么样呢?其实Go语言的标准库里面已经非常好的支持了JSON,可以对JSON包进行解析、生成JSON数据。 |
3 | | - |
4 | | -我们还是假设目前想要描述所有的服务器列表,通过JSON如何来表达,请看下面的描述 |
5 | | - |
6 | | - {"servers":[{"serverName":"Shanghai_VPN","serverIP":"127.0.0.1"},{"serverName":"Beijing_VPN","serverIP":"127.0.0.2"}]} |
7 | | - |
8 | | -接下来的例子以此JSON数据为基础,我们来进行JSON的解析和生成。 |
9 | | -##解析JSON |
10 | | - |
11 | | -###解析到结构体 |
12 | | -假如有了上面的JSON串,那么我们如何来解析这个JSON串呢?Go的JSON包中有如下函数 |
13 | | - |
14 | | - func Unmarshal(data []byte, v interface{}) error |
15 | | - |
16 | | -通过这个函数我们就可以实现解析的目的,详细的解析例子请看如下代码: |
17 | | - |
18 | | - package main |
19 | | - |
20 | | - import ( |
21 | | - "encoding/json" |
22 | | - "fmt" |
23 | | - ) |
24 | | - |
25 | | - type Server struct { |
26 | | - ServerName string |
27 | | - ServerIP string |
28 | | - } |
29 | | - |
30 | | - type Serverslice struct { |
31 | | - Servers []Server |
32 | | - } |
33 | | - |
34 | | - func main() { |
35 | | - var s Serverslice |
36 | | - str := `{"servers":[{"serverName":"Shanghai_VPN","serverIP":"127.0.0.1"},{"serverName":"Beijing_VPN","serverIP":"127.0.0.2"}]}` |
37 | | - json.Unmarshal([]byte(str), &s) |
38 | | - fmt.Println(s) |
39 | | - } |
40 | | - |
41 | | -通过上面的例子我们可以看到我们首先定义了结构体,结构体和JSON的数据一一对应,数组对应slice,字段名对应JSON里面的KEY,那么解析的时候如何解析到对应的字段的呢?例如JSON的key是`Foo`,那么怎么找对应的字段呢? |
42 | | - |
43 | | -- 首先查找字段的tag里面带有`Foo`的导出字段(首字母大写) |
44 | | -- 其次查找字段名是`Foo`的导出字段 |
45 | | -- 最后查找类似`FOO`或者`FoO`这样的除了首字母之外其他大小写不敏感的导出字段 |
46 | | - |
47 | | -聪明的你一定注意到了一点,能够输出的数据必须是导出字段,其他字段是不能输出的。同时JSON解析的时候只会解析能找得到的字段,如果找不到的字段会被忽略,这样的一个好处是在于当你接收到一个很大的JSON数据的时候,你如果只想部分数据,那么用这种方式就可以轻松的解决了。 |
48 | | - |
49 | | -###解析到interface |
50 | | -我们知道上面哪种解析方式是当我们了解了JSON的数据结构的情况下来进行的解析,那么如果我们在不知道JSON格式的情况下,如何来解析JSON呢? |
51 | | - |
52 | | -##生成JSON |
53 | | - |
54 | | -## links |
55 | | - * [目录](<preface.md>) |
56 | | - * 上一节: [XML处理](<7.1.md>) |
57 | | - * 下一节: [正则处理](<7.3.md>) |
58 | | - |
59 | | -## LastModified |
| 1 | +#7.2 JSON处理 |
| 2 | +JSON(Javascript Object Notation)是一种轻量级的数据交换语言,以文字为基础,具有自我描述性且易于让人阅读。尽管JSON是在Javascript的一个子集,但JSON是独立于语言的文本格式,并且采用了类似于C语言家族的一些习惯。JSON与XML最大的不同在于XML是一个完整的标记语言,而JSON不是。JSON由于比XML更小、更快,更易解析,以及浏览器的内建快速解析支持,使得其更适用于网络数据传输领域。目前我们看到很多的开放平台,基本上所有的接口都是采用了JSON作为他们的数据交互。那么JSON在Web开发中如此重要,Go语言对于JSON支持的怎么样呢?其实Go语言的标准库里面已经非常好的支持了JSON,可以对JSON包进行解析、生成JSON数据。 |
| 3 | + |
| 4 | +我们还是假设目前想要描述所有的服务器列表,通过JSON如何来表达,请看下面的描述 |
| 5 | + |
| 6 | + {"servers":[{"serverName":"Shanghai_VPN","serverIP":"127.0.0.1"},{"serverName":"Beijing_VPN","serverIP":"127.0.0.2"}]} |
| 7 | + |
| 8 | +接下来的例子以此JSON数据为基础,我们来进行JSON的解析和生成。 |
| 9 | +##解析JSON |
| 10 | + |
| 11 | +###解析到结构体 |
| 12 | +假如有了上面的JSON串,那么我们如何来解析这个JSON串呢?Go的JSON包中有如下函数 |
| 13 | + |
| 14 | + func Unmarshal(data []byte, v interface{}) error |
| 15 | + |
| 16 | +通过这个函数我们就可以实现解析的目的,详细的解析例子请看如下代码: |
| 17 | + |
| 18 | + package main |
| 19 | + |
| 20 | + import ( |
| 21 | + "encoding/json" |
| 22 | + "fmt" |
| 23 | + ) |
| 24 | + |
| 25 | + type Server struct { |
| 26 | + ServerName string |
| 27 | + ServerIP string |
| 28 | + } |
| 29 | + |
| 30 | + type Serverslice struct { |
| 31 | + Servers []Server |
| 32 | + } |
| 33 | + |
| 34 | + func main() { |
| 35 | + var s Serverslice |
| 36 | + str := `{"servers":[{"serverName":"Shanghai_VPN","serverIP":"127.0.0.1"},{"serverName":"Beijing_VPN","serverIP":"127.0.0.2"}]}` |
| 37 | + json.Unmarshal([]byte(str), &s) |
| 38 | + fmt.Println(s) |
| 39 | + } |
| 40 | + |
| 41 | +通过上面的例子我们可以看到我们首先定义了结构体,结构体和JSON的数据一一对应,数组对应slice,字段名对应JSON里面的KEY,那么解析的时候如何解析到对应的字段的呢?例如JSON的key是`Foo`,那么怎么找对应的字段呢? |
| 42 | + |
| 43 | +- 首先查找字段的tag里面带有`Foo`的导出字段(首字母大写) |
| 44 | +- 其次查找字段名是`Foo`的导出字段 |
| 45 | +- 最后查找类似`FOO`或者`FoO`这样的除了首字母之外其他大小写不敏感的导出字段 |
| 46 | + |
| 47 | +聪明的你一定注意到了一点,能够输出的数据必须是导出字段,其他字段是不能输出的。同时JSON解析的时候只会解析能找得到的字段,如果找不到的字段会被忽略,这样的一个好处是在于当你接收到一个很大的JSON数据的时候,你如果只想部分数据,那么用这种方式就可以轻松的解决了。 |
| 48 | + |
| 49 | +上面这个是官方提供的解决方案,其实很多时候我们通过类型断言,操作起来不是很方便,目前bitly公司开发了一个`simplejson`,在处理未知结构体的JSON处理中相当方便,详细例子如下所示: |
| 50 | + |
| 51 | + js, err := NewJson([]byte(`{ |
| 52 | + "test": { |
| 53 | + "array": [1, "2", 3], |
| 54 | + "int": 10, |
| 55 | + "float": 5.150, |
| 56 | + "bignum": 9223372036854775807, |
| 57 | + "string": "simplejson", |
| 58 | + "bool": true |
| 59 | + } |
| 60 | + }`)) |
| 61 | + |
| 62 | + arr, _ := js.Get("test").Get("array").Array() |
| 63 | + i, _ := js.Get("test").Get("int").Int() |
| 64 | + ms := js.Get("test").Get("string").MustString() |
| 65 | + |
| 66 | +我们看到通过这个库对于我们来说操作JSON非常的简单,比起前面介绍的官方方案更加简洁。 |
| 67 | + |
| 68 | +###解析到interface |
| 69 | +我们知道上面哪种解析方式是当我们了解了JSON的数据结构的情况下来进行的解析,那么如果我们在不知道JSON格式的情况下,如何来解析JSON呢? |
| 70 | + |
| 71 | +我们知道interface{}可以存储任意的数据类型,那么这正好符合JSON包在未知数据结构的时候来进行解析,JSON包中采用map[string]interface{}和[]interface{}结构来存储任意的JSON对象和数组。Go类型和JSON类型的对应关系如下: |
| 72 | + |
| 73 | +- bool 代表 JSON booleans, |
| 74 | +- float64 代表 JSON numbers, |
| 75 | +- string 代表 JSON strings, |
| 76 | +- nil 代表 JSON null. |
| 77 | + |
| 78 | +现在我们假设有如下的JSON数据 |
| 79 | + |
| 80 | + b := []byte(`{"Name":"Wednesday","Age":6,"Parents":["Gomez","Morticia"]}`) |
| 81 | + |
| 82 | +如果在我们不知道他的结构的情况下,我们把他解析到interface{}里面 |
| 83 | + |
| 84 | + var f interface{} |
| 85 | + err := json.Unmarshal(b, &f) |
| 86 | + |
| 87 | +这个时候f里面存储了一个map类似,他们的key是string,值存储在空的interface{}里 |
| 88 | + |
| 89 | + f = map[string]interface{}{ |
| 90 | + "Name": "Wednesday", |
| 91 | + "Age": 6, |
| 92 | + "Parents": []interface{}{ |
| 93 | + "Gomez", |
| 94 | + "Morticia", |
| 95 | + }, |
| 96 | + } |
| 97 | + |
| 98 | +那么如何来访问这些数据呢?通过断言的方式: |
| 99 | + |
| 100 | + m := f.(map[string]interface{}) |
| 101 | + |
| 102 | +通过断言之后,你就可以通过如下方式来访问里面的数据了 |
| 103 | + |
| 104 | + for k, v := range m { |
| 105 | + switch vv := v.(type) { |
| 106 | + case string: |
| 107 | + fmt.Println(k, "is string", vv) |
| 108 | + case int: |
| 109 | + fmt.Println(k, "is int", vv) |
| 110 | + case []interface{}: |
| 111 | + fmt.Println(k, "is an array:") |
| 112 | + for i, u := range vv { |
| 113 | + fmt.Println(i, u) |
| 114 | + } |
| 115 | + default: |
| 116 | + fmt.Println(k, "is of a type I don't know how to handle") |
| 117 | + } |
| 118 | + } |
| 119 | +通过上面的示例代码我们看到,我们就可以来访问结构不确定的JSON数据串。 |
| 120 | + |
| 121 | +##生成JSON |
| 122 | +我们开发很多应用的时候,最后都是要输出JSON数据串,那么如何来处理呢?JSON包里面通过`Marshal`函数来处理,函数定义如下: |
| 123 | + |
| 124 | + func Marshal(v interface{}) ([]byte, error) |
| 125 | + |
| 126 | +假设我们还是需要生成上面的服务器列表信息,那么如何来处理呢?请看下面的例子: |
| 127 | + |
| 128 | + package main |
| 129 | + |
| 130 | + import ( |
| 131 | + "encoding/json" |
| 132 | + "fmt" |
| 133 | + ) |
| 134 | + |
| 135 | + type Server struct { |
| 136 | + ServerName string |
| 137 | + ServerIP string |
| 138 | + } |
| 139 | + |
| 140 | + type Serverslice struct { |
| 141 | + Servers []Server |
| 142 | + } |
| 143 | + |
| 144 | + func main() { |
| 145 | + var s Serverslice |
| 146 | + s.Servers = append(s.Servers, Server{ServerName: "Shanghai_VPN", ServerIP: "127.0.0.1"}) |
| 147 | + s.Servers = append(s.Servers, Server{ServerName: "Beijing_VPN", ServerIP: "127.0.0.2"}) |
| 148 | + b, err := json.Marshal(s) |
| 149 | + if err != nil { |
| 150 | + fmt.Println("json err:", err) |
| 151 | + } |
| 152 | + fmt.Println(string(b)) |
| 153 | + } |
| 154 | + |
| 155 | +输出如下内容: |
| 156 | + |
| 157 | + {"Servers":[{"ServerName":"Shanghai_VPN","ServerIP":"127.0.0.1"},{"ServerName":"Beijing_VPN","ServerIP":"127.0.0.2"}]} |
| 158 | + |
| 159 | +我们看到上面的输出字段名都是大写的,如果你想用小写的怎么办呢?把结构体的字段名改成小写的?JSON输出的时候必须注意,只有导出的字段才会被输出,如果修改字段名,那么就会发现什么都不会输出,所以必须通过struct tag定义来实现: |
| 160 | + |
| 161 | + type Server struct { |
| 162 | + ServerName string `json:"serverName"` |
| 163 | + ServerIP string `json:"serverIP"` |
| 164 | + } |
| 165 | + |
| 166 | + type Serverslice struct { |
| 167 | + Servers []Server `json:"servers"` |
| 168 | + } |
| 169 | + |
| 170 | +通过修改上面的结构体定义,这样输出的JSON串和我们最开始定义的JSON串就保持一模一样了。 |
| 171 | + |
| 172 | +针对JSON的输出我们struct tag的定义需要注意一下几点设置: |
| 173 | + |
| 174 | +- tag中带有`"-"`,那么这个字段不会输出到JSON |
| 175 | +- tag中带有自定义名称,那么这个自定义名称会出现在JSON的字段名中,例如上面例子中serverName |
| 176 | +- tag中如果带有`"omitempty"`,那么如果该字段值为空,就不会输出到JSON串中 |
| 177 | +- 如果字段类型是int,而tag中带有`"string"`,那么这个字段在输出到JSON的时候会把该字段对应的值转换成JSON字符串 |
| 178 | + |
| 179 | +Marshal函数只有在转换成功的时候才会返回数据,在转换的过程中我们需要注意几点: |
| 180 | + |
| 181 | +- JSON对象只支持string作为key,所以要编码一个map,那么必须是map[string]T这种类型(T是Go语言中任意的类型) |
| 182 | +- Channel, complex和function是不能被编码成JSON的 |
| 183 | +- 嵌套的数据是不能编码的,不然会让JSON编码进入死循环 |
| 184 | +- 指针在编码的时候会输出指针指向的内容,而空指针会输出null |
| 185 | + |
| 186 | + |
| 187 | +通过上面这个讲解,我们了解了如何使用Go语言的标准包里面的JSON来进行编解JSON数据,同时介绍了第三方包`go-simplejson`如何方便的操作JSON数据,这个对于我们接下来的Web开发相当重要。 |
| 188 | + |
| 189 | +## links |
| 190 | + * [目录](<preface.md>) |
| 191 | + * 上一节: [XML处理](<7.1.md>) |
| 192 | + * 下一节: [正则处理](<7.3.md>) |
| 193 | + |
| 194 | +## LastModified |
60 | 195 | * $Id$ |
0 commit comments