golang json用法講解


 

簡介

json格式可以算我們日常最常用的序列化格式之一了,Go語言作為一個由Google開發,號稱互聯網的C語言的語言,自然也對JSON格式支持很好。但是Go語言是個強類型語言,對格式要求極其嚴格而JSON格式雖然也有類型,但是並不穩定,Go語言在解析來源為非強類型語言時比如PHP等序列化的JSON時,經常遇到一些問題諸如字段類型變化導致無法正常解析的情況,導致服務不穩定。所以本篇的主要目的

  1. 就是挖掘Golang解析json的絕大部分能力
  2. 比較優雅的解決解析json時存在的各種問題
  3. 深入一下Golang解析json的過程
  • Golang解析JSON之Tag篇

  1. 一個結構體正常序列化過后是什么樣的呢?

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
     ProductID int64
     Number    int
     Price     float64
     IsOnSale  bool
}
 
func  main() {
     p := &Product{}
     p.Name =  "Xiao mi 6"
     p.IsOnSale = true
     p.Number = 10000
     p.Price = 2499.00
     p.ProductID = 1
     data, _ := json.Marshal(p)
     fmt.Println(string(data))
}
 
 
//結果
{ "Name" : "Xiao mi 6" , "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" `
     ProductID int64   `json: "-" // 表示不進行序列化
     Number    int     `json: "number" `
     Price     float64 `json: "price" `
     IsOnSale  bool    `json: "is_on_sale,string" `
}
 
// 序列化過后,可以看見
    { "name" : "Xiao mi 6" , "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" `
     ProductID int64   `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 =  "Xiao mi 6"
     p.IsOnSale = false
     p.Number = 10000
     p.Price = 2499.00
     p.ProductID = 0
 
     data, _ := json.Marshal(p)
     fmt.Println(string(data))
}
// 結果
{ "name" : "Xiao mi 6" , "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" `
     ProductID int64   `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" : "Xiao mi 6" , "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>
{Xiao mi 6 10 10000 2499 true}
  • 下面講一講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  {
     Value string
}
 
// UnmarshalJSON _
func  (m *Mail) UnmarshalJSON(data []byte) error {
     // 這里簡單演示一下,簡單判斷即可
     if  bytes.Contains(data, []byte( "@" )) {
         return  fmt.Errorf( "mail format error" )
     }
     m.Value = string(data)
     return  nil
}
 
// UnmarshalJSON _
func  (m *Mail) MarshalJSON() (data []byte, err error) {
     if  m != nil {
         data = []byte(m.Value)
     }
     return
}
 
// Phone _
type  Phone  struct  {
     Value string
}
 
// UnmarshalJSON _
func  (p *Phone) UnmarshalJSON(data []byte) error {
     // 這里簡單演示一下,簡單判斷即可
     if  len(data) != 11 {
         return  fmt.Errorf( "phone format error" )
     }
     p.Value = string(data)
     return  nil
}
 
// UnmarshalJSON _
func  (p *Phone) MarshalJSON() (data []byte, err error) {
     if  p != nil {
         data = []byte(p.Value)
     }
     return
}
 
// UserRequest _
type  UserRequest  struct  {
     Name  string
     Mail  Mail
     Phone Phone
}
 
func  main() {
     user := UserRequest{}
     user.Name =  "ysy"
     user.Mail.Value =  "yangshiyu@x.com"
     user.Phone.Value =  "18900001111"
     fmt.Println(json.Marshal(user))
}

  

為什么要這樣?

如果是客戶端開發,需要開發大量的API,接收大量的JSON,在開發早期定義各種類型看起來是很大的工作量,不如寫 if else 判斷數據簡單暴力。但是到開發末期,你會發現預先定義的方式能極大的提高你的代碼質量,減少代碼量。下面實例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
     Phone string
}
func  AddUser(data []byte) (err error) {
     user := &UserRequest{}
     err = json.Unmarshal(data, user)
     if  err != nil {
         return
     }
     // 
     if  isMail(user.Mail) {
         return  fmt.Errorf( "mail format error"
     }
 
     if  isPhone(user.Phone) {
         return  fmt.Errorf( "phone format error" )
     }
 
     // TODO
     return
 
實例2,利用預先定義好的類型,在解析時就進行判斷
// UserRequest _
type  UserRequest  struct  {
     Name  string
     Mail  Mail
     Phone Phone
}
 
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

 
 


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM