go 处理json
在云原生的相关技术中, 调用 kafka
和 es
等中间件,返回的结果大多是 json
格式,因此我们需要频繁的对 json
进行操作。
在go
这种强类型的语言中,不同于 python
和 nodejs
等弱类型的语言,对json
取值比较麻烦,一般有三种方法
-
把
json
映射为 map 格式 -
把
json
映射为stuct
格式 -
借助于
第三方库
,直接对json
对象取值
三种方案的实例
package main
import (
"encoding/json"
"fmt"
"github.com/tidwall/gjson"
)
func main() {
byt := []byte(`{
"num":6.13,
"strs":["a","b"],
"obj":{"foo":{"bar":"zip","zap":6}}
}`)
fmt.Println("func1**************************************")
jsonByMap(byt)
fmt.Println("func2**************************************")
jsonByStruct(byt)
fmt.Println("func3**************************************")
jsonBygjson(byt)
}
func jsonByMap(byt []byte) {
var dat map[string]interface{}
if err := json.Unmarshal(byt, &dat); err != nil {
panic(err)
}
//fmt.Println(dat)
num := dat["num"].(float64)
fmt.Println(num)
strs := dat["strs"].([]interface{})
str1 := strs[0].(string)
fmt.Println(str1)
obj := dat["obj"].(map[string]interface{})
obj2 := obj["foo"].(map[string]interface{})
fmt.Println(obj2)
}
func jsonByStruct(byt []byte) {
type ourData struct {
Num float64 `json:"num"`
Strs []string `json:"strs"`
Obj map[string]map[string]string `json:"obj"`
}
res := ourData{}
json.Unmarshal(byt, &res)
fmt.Println(res.Num)
fmt.Println(res.Strs)
fmt.Println(res.Obj)
}
func jsonBygjson(byt []byte) {
//首先校验 json库是否合法
str := string(byt)
if !gjson.Valid(str) {
fmt.Println("json字符串不合法")
return
}
//从json中取string
value := gjson.Get(str, "obj.foo.bar")
fmt.Println(value.String())
//从json中取int
num := gjson.Get(str, "num")
fmt.Println(num.Int())
//json中取数组
strs := gjson.Get(str, "strs")
if strs.IsArray() {
for _, v := range strs.Array() {
fmt.Println("数组", v.String())
}
}
//json的链式调用
foo := gjson.Get(str, "obj.foo")
fmt.Println(foo.Get("zap"))
}
以上的三种方法中,第一种映射为字典格式的方法,需要对每一种类型进行断言,这个过程比较麻烦。第二种方法,要先声明一个结构体,然而在真实的应用场景中,结构体的字段往往较多,写起来会很麻烦。所以本文建议使用第三方库的方法。关于 gjson的更多详细用法,可以参考文献3
查考文献
https://mp.weixin.qq.com/s/0OcL7lx189zKAen4Al8J5A
https://www.bilibili.com/read/cv12373667
https://blog.csdn.net/ma_jiang/article/details/120719629