golang對(duì)json序列化和反序列化的操作實(shí)在是難受墅茉,所以說(shuō)用習(xí)慣了高級(jí)語(yǔ)言特性伟阔,再轉(zhuǎn)到這些偏原生的寫法上就會(huì)很難受搭综。
不多BB,開(kāi)始記錄站辉。
序列化庫(kù)的選擇
當(dāng)寫個(gè)小demo或者做個(gè)小工具呢撞,沒(méi)有大規(guī)模使用場(chǎng)景损姜,那使用哪個(gè)庫(kù)都是一樣的,因?yàn)樾阅艿捏w現(xiàn)并不會(huì)很明顯殊霞。但是如果是在實(shí)際項(xiàng)目中使用摧阅,且伴隨著高并發(fā),大容量等場(chǎng)景脓鹃,我還是推薦使用json-iterator
逸尖。
go自帶json庫(kù)
"encoding/json" 官當(dāng)自帶
json-iterator
號(hào)稱最快的go json解析器。跟官方的寫法兼容瘸右,我目前基本都使用這個(gè)娇跟。
https://github.com/json-iterator/go
效率對(duì)比
ns 納秒 op 操作
ns/op | allocation bytes | allocation times | |
---|---|---|---|
std decode | 35510 ns/op | 1960 B/op | 99 allocs/op |
easyjson decode | 8499 ns/op | 160 B/op | 4 allocs/op |
jsoniter decode | 5623 ns/op | 160 B/op | 3 allocs/op |
std encode | 2213 ns/op | 712 B/op | 5 allocs/op |
easyjson encode | 883 ns/op | 576 B/op | 3 allocs/op |
jsoniter encode | 837 ns/op | 384 B/op | 4 allocs/op |
編碼案例
type Hero struct {
Name string
Age int
Birthday string
Sal float64
Skill string
}
序列化
hero := Hero{
Name: "小王",
Age: 20,
Birthday: "2021-02-23",
Sal: 88.02,
Skill: "技能",
}
jsonStu, err := json.Marshal
if err != nil {
fmt.Println("生成json字
}
fmt.Println(string(jsonStu))
反序列化
結(jié)構(gòu)體 struct
str := "{\"Name\":\"張三豐\",\"Age\":98,\"Birthday\":\"2001-09-21\",\"Sal\":3800.85,\"Skill\":\"武當(dāng)劍法\"}"
var hero Hero
err := json.Unmarshal([]byte(str), &hero)
if err != nil {
fmt.Printf("unmarshal err=%v\n", err)
}
結(jié)構(gòu)體數(shù)組
倆種方式,一種直接反序列化成 結(jié)構(gòu)體數(shù)組太颤,另一種反序列化為 slice苞俘,內(nèi)容為map[string]interface{}
結(jié)構(gòu)體數(shù)組
str := `[{"Name":"張三豐","Age":98,"Birthday":"2001-09-21","Sal":3800.85,"Skill":"武當(dāng)劍法"},{"Name":"張無(wú)忌","Age":28,"Birthday":"2004-09-21","Sal":300.85,"Skill":"乾坤大挪移"}]`
var hero []Hero
err := json.Unmarshal([]byte(str), &hero)
if err != nil {
fmt.Printf("unmarshal err=%v\n", err)
}
fmt.Printf("反序列化后 hero=%v", hero)
slice
str := `[{"Name":"張三豐","Age":98,"Birthday":"2001-09-21","Sal":3800.85,"Skill":"武當(dāng)劍法"},{"Name":"張無(wú)忌","Age":28,"Birthday":"2004-09-21","Sal":300.85,"Skill":"乾坤大挪移"}]`
//定義一個(gè)slice
var slice []map[string]interface{}
//注意:反序列化map,不需要make,因?yàn)閙ake操作被封裝到Unmarshal函數(shù)
err := json.Unmarshal([]byte(str), &slice)
if err != nil {
fmt.Printf("unmarshal err=%v\n", err)
}
fmt.Printf("反序列化后 slice=%v\n", slice)