golang json用法讲解
本文内容纲要:
-
-简介
-Golang解析JSON之Tag篇
-下面讲一讲Golang如何自定义解析JSON,Golang自带的JSON解析功能非常强悍
-为什么要这样?
简介
json格式可以算我们日常最常用的序列化格式之一了,Go语言作为一个由Google开发,号称互联网的C语言的语言,自然也对JSON格式支持很好。但是Go语言是个强类型语言,对格式要求极其严格而JSON格式虽然也有类型,但是并不稳定,Go语言在解析来源为非强类型语言时比如PHP等序列化的JSON时,经常遇到一些问题诸如字段类型变化导致无法正常解析的情况,导致服务不稳定。所以本篇的主要目的
- 就是挖掘Golang解析json的绝大部分能力
- 比较优雅的解决解析json时存在的各种问题
- 深入一下Golang解析json的过程
-
Golang解析JSON之Tag篇
- 一个结构体正常序列化过后是什么样的呢?
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
|
package main
import (
"encoding/json"
"fmt"
)
//Product商品信息
type Product struct {
Name string
ProductIDint64
Number int
Price float64
IsOnSale bool
}
func main(){
p:=&Product{}
p.Name= "Xiaomi6"
p.IsOnSale=true
p.Number=10000
p.Price=2499.00
p.ProductID=1
data,_:=json.Marshal(p)
fmt.Println(string(data))
}
//结果
{ "Name" : "Xiaomi6" , "ProductID" :1, "Number" :10000, "Price" :2499, "IsOnSale" :true}
|
2.何为Tag,tag就是标签,给结构体的每个字段打上一个标签,标签冒号前是类型,后面是标签名
1
2
3
4
5
6
7
8
9
10
11
|
//Product_
type Product struct {
Name string `json: "name" `
ProductIDint64 `json: "-" ` //表示不进行序列化
Number int `json: "number" `
Price float64`json: "price" `
IsOnSale bool `json: "is_on_sale,string" `
}
//序列化过后,可以看见
{ "name" : "Xiaomi6" , "number" :10000, "price" :2499, "is_on_sale" : "false" }
|
3.omitempty,tag里面加上omitempy,可以在序列化的时候忽略0值或者空值
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
|
package main
import (
"encoding/json"
"fmt"
)
//Product_
type Product struct {
Name string `json: "name" `
ProductIDint64 `json: "product_id,omitempty" `
Number int `json: "number" `
Price float64`json: "price" `
IsOnSale bool `json: "is_on_sale,omitempty" `
}
func main(){
p:=&Product{}
p.Name= "Xiaomi6"
p.IsOnSale=false
p.Number=10000
p.Price=2499.00
p.ProductID=0
data,_:=json.Marshal(p)
fmt.Println(string(data))
}
//结果
{ "name" : "Xiaomi6" , "number" :10000, "price" :2499}
|
4.type,有些时候,我们在序列化或者反序列化的时候,可能结构体类型和需要的类型不一致,这个时候可以指定,支持string,number和boolean
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
|
package main
import (
"encoding/json"
"fmt"
)
//Product_
type Product struct {
Name string `json: "name" `
ProductIDint64 `json: "product_id,string" `
Number int `json: "number,string" `
Price float64`json: "price,string" `
IsOnSale bool `json: "is_on_sale,string" `
}
func main(){
var data=`{ "name" : "Xiaomi6" , "product_id" : "10" , "number" : "10000" , "price" : "2499" , "is_on_sale" : "true" }`
p:=&Product{}
err:=json.Unmarshal([]byte(data),p)
fmt.Println(err)
fmt.Println(*p)
}
//结果
<nil>
{Xiaomi610100002499true}
|
-
下面讲一讲Golang如何自定义解析JSON,Golang自带的JSON解析功能非常强悍
说明
很多时候,我们可能遇到这样的场景,就是远端返回的JSON数据不是你想要的类型,或者你想做额外的操作,比如在解析的过程中进行校验,或者类型转换,那么我们可以这样或者在解析过程中进行数据转换
实例
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
|
package main
import (
"bytes"
"encoding/json"
"fmt"
)
//Mail_
type Mail struct {
Valuestring
}
//UnmarshalJSON_
func (m*Mail)UnmarshalJSON(data[]byte)error{
//这里简单演示一下,简单判断即可
if bytes.Contains(data,[]byte( "@" )){
return fmt.Errorf( "mailformaterror" )
}
m.Value=string(data)
return nil
}
//UnmarshalJSON_
func (m*Mail)MarshalJSON()(data[]byte,errerror){
if m!=nil{
data=[]byte(m.Value)
}
return
}
//Phone_
type Phone struct {
Valuestring
}
//UnmarshalJSON_
func (p*Phone)UnmarshalJSON(data[]byte)error{
//这里简单演示一下,简单判断即可
if len(data)!=11{
return fmt.Errorf( "phoneformaterror" )
}
p.Value=string(data)
return nil
}
//UnmarshalJSON_
func (p*Phone)MarshalJSON()(data[]byte,errerror){
if p!=nil{
data=[]byte(p.Value)
}
return
}
//UserRequest_
type UserRequest struct {
Name string
Mail Mail
PhonePhone
}
func main(){
user:=UserRequest{}
user.Name= "ysy"
user.Mail.Value= "yangshiyu@x.com"
user.Phone.Value= "18900001111"
fmt.Println(json.Marshal(user))
}
|
为什么要这样?
如果是客户端开发,需要开发大量的API,接收大量的JSON,在开发早期定义各种类型看起来是很大的工作量,不如写ifelse判断数据简单暴力。但是到开发末期,你会发现预先定义的方式能极大的提高你的代码质量,减少代码量。下面实例1和实例2,谁能减少代码一目了然
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
|
实例1, if else 做数据校验
//UserRequest_
type UserRequest struct {
Name string
Mail string
Phonestring
}
func AddUser(data[]byte)(errerror){
user:=&UserRequest{}
err=json.Unmarshal(data,user)
if err!=nil{
return
}
//
if isMail(user.Mail){
return fmt.Errorf( "mailformaterror" )
}
if isPhone(user.Phone){
return fmt.Errorf( "phoneformaterror" )
}
//TODO
return
}
实例2,利用预先定义好的类型,在解析时就进行判断
//UserRequest_
type UserRequest struct {
Name string
Mail Mail
PhonePhone
}
func AddUser(data[]byte){
user:=&UserRequest{}
err=json.Unmarshal(data,user)
if err!=nil{
return
}
//TODO
}
|
转自:http://www.cnblogs.com/yangshiyu/p/6942414.html
https://www.cnblogs.com/52php/p/6518728.html
本文内容总结:,简介,Golang解析JSON之Tag篇,下面讲一讲Golang如何自定义解析JSON,Golang自带的JSON解析功能非常强悍,为什么要这样?,
原文链接:https://www.cnblogs.com/williamjie/p/9927281.html