前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >Go中解析JSON

Go中解析JSON

原创
作者头像
ruochen
发布2021-12-05 23:06:18
3.1K0
发布2021-12-05 23:06:18
举报
安装

要使用gjson,先要安装go环境并执行go get:

代码语言:txt
复制
$ go get -u github.com/tidwall/gjson

以上命令会检索并下载该库到Go环境中。

Get函数获取值

Get在json中搜索指定的路径。路径用点语法表示,比如“name.last"或“age"。这个函数需要提供格式正规和有效的json值。无效的json不会引起panic,但它可能返回意外的结果。当找到值后立即返回。

代码语言:txt
复制
package main
代码语言:txt
复制
import "github.com/tidwall/gjson"
代码语言:txt
复制
const json = `{"name":{"first":"Janet","last":"Prichard"},"age":47}`
代码语言:txt
复制
func main() {
代码语言:txt
复制
    result := gjson.Get(json, "name.last")
代码语言:txt
复制
    println(result.String())
代码语言:txt
复制
}

output

代码语言:txt
复制
Prichard

还有用于处理JSON字节切片的GetBytes(https://links.jianshu.com/go?to=https%3A%2F%2Fwww.ctolib.com%2Fgjson.html%23working-

with-bytes) 函数。

path语法

path是由点分隔的一系列键。key可以包含特殊的通配符'*'和'?'。要访问数组值,请使用索引作为键。要获取数组中的元素数量或访问子路径,请使用'#'字符。点和通配符可以用'\'转义。

代码语言:txt
复制
{
代码语言:txt
复制
  "name": {"first": "Tom", "last": "Anderson"},
代码语言:txt
复制
  "age":37,
代码语言:txt
复制
  "children": ["Sara","Alex","Jack"],
代码语言:txt
复制
  "fav.movie": "Deer Hunter",
代码语言:txt
复制
  "friends": [
代码语言:txt
复制
    {"first": "James", "last": "Murphy"},
代码语言:txt
复制
    {"first": "Roger", "last": "Craig"}
代码语言:txt
复制
  ]
代码语言:txt
复制
}
代码语言:txt
复制
"name.last"          >> "Anderson"
代码语言:txt
复制
"age"                >> 37
代码语言:txt
复制
"children"           >> ["Sara","Alex","Jack"]
代码语言:txt
复制
"children.#"         >> 3
代码语言:txt
复制
"children.1"         >> "Alex"
代码语言:txt
复制
"child*.2"           >> "Jack"
代码语言:txt
复制
"c?ildren.0"         >> "Sara"
代码语言:txt
复制
"fav\.movie"         >> "Deer Hunter"
代码语言:txt
复制
"friends.#.first"    >> ["James","Roger"]
代码语言:txt
复制
"friends.1.last"     >> "Craig"

查询一个数组:

代码语言:txt
复制
`friends.#[last="Murphy"].first` >> "James"
结果类型

GJSON支持json类型字符串,数字,bool和null。数组和对象作为原始json类型返回。

Result类型包含以下类型之一:

代码语言:txt
复制
bool, for JSON booleans
代码语言:txt
复制
float64, for JSON numbers
代码语言:txt
复制
string, for JSON string literals
代码语言:txt
复制
nil, for JSON null

要直接访问该值:

代码语言:txt
复制
result.Type    // 可能是String, Number, True, False, Null, or JSON
代码语言:txt
复制
result.Str     // 保存string
代码语言:txt
复制
result.Num     // 保存float64
代码语言:txt
复制
result.Raw     // 保存 raw json
代码语言:txt
复制
result.Index   // json中原始值的索引,0表示索引未知

result有很多函数:

代码语言:txt
复制
result.Value() interface{}
代码语言:txt
复制
result.Int() int64
代码语言:txt
复制
result.Uint() uint64
代码语言:txt
复制
result.Float() float64
代码语言:txt
复制
result.String() string
代码语言:txt
复制
result.Bool() bool
代码语言:txt
复制
result.Array() []gjson.Result
代码语言:txt
复制
result.Map() map[string]gjson.Result
代码语言:txt
复制
result.Get(path string) Result

result.Value返回一个interface{},需要断言,是以下类型之一:

代码语言:txt
复制
boolean >> bool
代码语言:txt
复制
number  >> float64
代码语言:txt
复制
string  >> string
代码语言:txt
复制
null    >> nil
代码语言:txt
复制
array   >> []interface{}
代码语言:txt
复制
object  >> map[string]interface{}

result.Array()返回一个数组。如果result代表一个不存在的值,那么将返回一个空数组。如果result不是一个JSON数组,返回值将是一个包含一个结果的数组。

Get获取嵌套数组值

假设你想要以下json中的所有lastName对应值:

代码语言:txt
复制
{
代码语言:txt
复制
  "programmers": [
代码语言:txt
复制
    {
代码语言:txt
复制
      "firstName": "Janet", 
代码语言:txt
复制
      "lastName": "McLaughlin", 
代码语言:txt
复制
    }, {
代码语言:txt
复制
      "firstName": "Elliotte", 
代码语言:txt
复制
      "lastName": "Hunter", 
代码语言:txt
复制
    }, {
代码语言:txt
复制
      "firstName": "Jason", 
代码语言:txt
复制
      "lastName": "Harold", 
代码语言:txt
复制
    }
代码语言:txt
复制
  ]
代码语言:txt
复制
}

您将使用路径“programmers.#.lastName“像这样:

代码语言:txt
复制
result := gjson.Get(json, "programmers.#.lastName")
代码语言:txt
复制
for _,name := range result.Array() {
代码语言:txt
复制
    println(name.String())
代码语言:txt
复制
}

你也可以查询数组中的对象:

代码语言:txt
复制
name := gjson.Get(json, `programmers.#[lastName="Hunter"].firstName`)
代码语言:txt
复制
println(name.String())  // 输出 "Elliotte"
parse和Get

有一个Parse(json)函数将执行简单的解析,result.Get(path)将搜索结果。例如,以下代码都将返回相同的结果:

代码语言:txt
复制
gjson.Parse(json).Get("name").Get("last")
代码语言:txt
复制
gjson.Get(json, "name").Get("last")
代码语言:txt
复制
gjson.Get(json, "name.last")
检查值是否存在

有时你只是想知道一个值是否存在。

代码语言:txt
复制
value := gjson.Get(json, "name.last")
代码语言:txt
复制
if !value.Exists() {
代码语言:txt
复制
    println("no last name")
代码语言:txt
复制
} else {
代码语言:txt
复制
    println(value.String())
代码语言:txt
复制
}
代码语言:txt
复制
// 或一步到位
代码语言:txt
复制
if gjson.Get(json, "name.last").Exists(){
代码语言:txt
复制
    println("has a last name")
代码语言:txt
复制
}
反序列化到map
代码语言:txt
复制
m, ok := gjson.Parse(json).Value().(map[string]interface{})
代码语言:txt
复制
if !ok{
代码语言:txt
复制
    // 不是map
代码语言:txt
复制
}
使用字节

如果json存在[]byte切片中,有一个

GetBytes

函数。首选使用:Get(string(data), path)

代码语言:txt
复制
var json []byte = ...
代码语言:txt
复制
result := gjson.GetBytes(json, path)

如果你使用gjson.GetBytes(json, path)函数,想避免将result.raw转换到[]byte,可以使用如下模式:

代码语言:txt
复制
var json []byte = ...
代码语言:txt
复制
result := gjson.GetBytes(json, path)
代码语言:txt
复制
var raw []byte
代码语言:txt
复制
if result.Index > 0 {
代码语言:txt
复制
    raw = json[result.Index:result.Index+len(result.Raw)]
代码语言:txt
复制
} else {
代码语言:txt
复制
    raw = []byte(result.Raw)
代码语言:txt
复制
}

这种方式不为原json分配子切片。这个方法使用result.Index属性,这是原始数据在原始json中的位置。result.Index的值可能等于0,这种情况下result.Raw被转成[]byte。

性能

与encoding/json, ffjson, EasyJSON和jsonparser,对比 GJSON的基准测试:

代码语言:txt
复制
BenchmarkGJSONGet-8                 15000000           333 ns/op           0 B/op          0 allocs/op
代码语言:txt
复制
BenchmarkGJSONUnmarshalMap-8          900000          4188 ns/op        1920 B/op         26 allocs/op
代码语言:txt
复制
BenchmarkJSONUnmarshalMap-8           600000          8908 ns/op        3048 B/op         69 allocs/op
代码语言:txt
复制
BenchmarkJSONUnmarshalStruct-8        600000          9026 ns/op        1832 B/op         69 allocs/op
代码语言:txt
复制
BenchmarkJSONDecoder-8                300000         14339 ns/op        4224 B/op        184 allocs/op
代码语言:txt
复制
BenchmarkFFJSONLexer-8               1500000          3156 ns/op         896 B/op          8 allocs/op
代码语言:txt
复制
BenchmarkEasyJSONLexer-8             3000000           938 ns/op         613 B/op          6 allocs/op
代码语言:txt
复制
BenchmarkJSONParserGet-8             3000000           442 ns/op          21 B/op          0 allocs/op

使用的json文档:

代码语言:txt
复制
{
代码语言:txt
复制
  "widget": {
代码语言:txt
复制
    "debug": "on",
代码语言:txt
复制
    "window": {
代码语言:txt
复制
      "title": "Sample Konfabulator Widget",
代码语言:txt
复制
      "name": "main_window",
代码语言:txt
复制
      "width": 500,
代码语言:txt
复制
      "height": 500
代码语言:txt
复制
    },
代码语言:txt
复制
    "image": { 
代码语言:txt
复制
      "src": "Images/Sun.png",
代码语言:txt
复制
      "hOffset": 250,
代码语言:txt
复制
      "vOffset": 250,
代码语言:txt
复制
      "alignment": "center"
代码语言:txt
复制
    },
代码语言:txt
复制
    "text": {
代码语言:txt
复制
      "data": "Click Here",
代码语言:txt
复制
      "size": 36,
代码语言:txt
复制
      "style": "bold",
代码语言:txt
复制
      "vOffset": 100,
代码语言:txt
复制
      "alignment": "center",
代码语言:txt
复制
      "onMouseUp": "sun1.opacity = (sun1.opacity / 100) * 90;"
代码语言:txt
复制
    }
代码语言:txt
复制
  }
代码语言:txt
复制
}    

执行的搜索操作:

代码语言:txt
复制
widget.window.name
代码语言:txt
复制
widget.image.hOffset
代码语言:txt
复制
widget.text.onMouseUp

原创声明:本文系作者授权腾讯云开发者社区发表,未经许可,不得转载。

如有侵权,请联系 cloudcommunity@tencent.com 删除。

原创声明:本文系作者授权腾讯云开发者社区发表,未经许可,不得转载。

如有侵权,请联系 cloudcommunity@tencent.com 删除。

评论
作者已关闭评论
0 条评论
热度
最新
推荐阅读
目录
  • 安装
  • Get函数获取值
  • path语法
  • 结果类型
  • Get获取嵌套数组值
  • parse和Get
  • 检查值是否存在
    • 反序列化到map
    • 使用字节
    • 性能
    领券
    问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档