程序员社区

gjson:在Go中快速解析JSON

gjson:在Go中快速解析JSON插图

GJSON是一个Go包,它提供了一种非常快速和简单的方式从json文档中获取值。这个库的目的是为BuntDB 项目提供高效的json索引。

点击链接 JSONed查看命令行接口。

安装

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

$ go get -u github.com/tidwall/gjson

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

Get函数获取值

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

package main

import "github.com/tidwall/gjson"

const json = `{"name":{"first":"Janet","last":"Prichard"},"age":47}`

func main() {
    result := gjson.Get(json, "name.last")
    println(result.String())
}

output

Prichard

还有用于处理JSON字节切片的GetBytes 函数。

path语法

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

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

查询一个数组:

`friends.#[last="Murphy"].first` >> "James"

结果类型

GJSON支持json类型字符串,数字,bool和null。数组和对象作为原始json类型返回。
Result类型包含以下类型之一:

bool, for JSON booleans
float64, for JSON numbers
string, for JSON string literals
nil, for JSON null

要直接访问该值:

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

result有很多函数:

result.Value() interface{}
result.Int() int64
result.Uint() uint64
result.Float() float64
result.String() string
result.Bool() bool
result.Array() []gjson.Result
result.Map() map[string]gjson.Result
result.Get(path string) Result

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

boolean >> bool
number  >> float64
string  >> string
null    >> nil
array   >> []interface{}
object  >> map[string]interface{}

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

Get获取嵌套数组值

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

{
  "programmers": [
    {
      "firstName": "Janet", 
      "lastName": "McLaughlin", 
    }, {
      "firstName": "Elliotte", 
      "lastName": "Hunter", 
    }, {
      "firstName": "Jason", 
      "lastName": "Harold", 
    }
  ]
}

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

result := gjson.Get(json, "programmers.#.lastName")
for _,name := range result.Array() {
    println(name.String())
}

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

name := gjson.Get(json, `programmers.#[lastName="Hunter"].firstName`)
println(name.String())  // 输出 "Elliotte"

parse和Get

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

gjson.Parse(json).Get("name").Get("last")
gjson.Get(json, "name").Get("last")
gjson.Get(json, "name.last")

检查值是否存在

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

value := gjson.Get(json, "name.last")
if !value.Exists() {
    println("no last name")
} else {
    println(value.String())
}

// 或一步到位
if gjson.Get(json, "name.last").Exists(){
    println("has a last name")
}
反序列化到map
m, ok := gjson.Parse(json).Value().(map[string]interface{})
if !ok{
    // 不是map
}

使用字节

如果json存在[]byte切片中,有一个 GetBytes 函数。首选使用:Get(string(data), path)

var json []byte = ...
result := gjson.GetBytes(json, path)

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

var json []byte = ...
result := gjson.GetBytes(json, path)
var raw []byte
if result.Index > 0 {
    raw = json[result.Index:result.Index+len(result.Raw)]
} else {
    raw = []byte(result.Raw)
}

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

性能

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

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

使用的json文档:

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

执行的搜索操作:

widget.window.name
widget.image.hOffset
widget.text.onMouseUp
赞(0) 打赏
未经允许不得转载:IDEA激活码 » gjson:在Go中快速解析JSON

一个分享Java & Python知识的社区