Go--gjson
GJSON 是一个用于处理 JSON 数据的 Go 语言库。它提供了一些方便的功能,例如解析 JSON 字符串、查询 JSON 对象、生成 JSON 对象等
下载gjson: go get -u github.com/tidwall/gjson
1、使用
传入 JSON 串和要读取的键路径,路径使用点号语法,如 "name.last "或 "age"。一旦找到值,就会立即返回。
路径是一系列用点分隔的键。键可以包含特殊的通配符 "*"和"? 要访问数组值,使用索引作为键。要获取数组中的元素个数或访问子路径,请使用 "#"字符。点和通配符可以用'\'转义。
例:
package main import (
"fmt"
"github.com/tidwall/gjson"
) func main() {
//{
// "name": {"first": "Tom", "last": "Anderson"},
// "age":37,
// "children": ["Sara","Alex","Jack"],
// "friends": [
// {"first": "James", "last": "Murphy"},
// {"first": "Roger", "last": "Craig"}
// ]
// }
//设置json字符串
jsonStr := `{"name": {"first": "Tom", "last": "Anderson"}, "age": 37, "children": ["Sara", "Alex", "Jack"], "friends": [{"first": "James", "last": "Murphy"}, {"first": "Roger", "last": "Craig"}]}` //获取数据
name1 := gjson.Get(jsonStr, "name") //默认返回的是gjson.Result类型,如果是直接打印,可直接使用,若是需进一步操作,建立转换成对应类型
name2 := gjson.Get(jsonStr, "name").String() //从 jsonStr 中获取 "name" 字段的值,然后通过调用 .String() 方法,将获取到的值转换为字符串类型,如果 "name" 字段的值不是字符串类型,它会被转换为字符串
name3 := gjson.Get(jsonStr, "name").Str //也是从 jsonStr 中获取 "name" 字段的值,但是是直接获取该字段的值,没有进行类型转换,如果"name"字段的值不是字符串类型,则将保持原来的类型
//如果确定 "name" 字段的值一定是字符串类型,那么使用 .Str 是更简洁的方式。如果不确定 "name" 字段的值是什么类型,或者想要确保它是一个字符串类型,那么使用 .String() 进行显式转换是更安全的选择。 fmt.Printf("值类型:%T,值:%v\n", name1, name1)
fmt.Printf("值类型:%T,值:%v\n", name2, name2)
fmt.Printf("值类型:%T,值:%v\n", name3, name3) //无法输出值是因为name" 是一个嵌套的 JSON 对象,而不是一个字符串
first := gjson.Get(jsonStr, "name.first").Str //需要进一步获取值
fmt.Printf("值类型:%T,值:%v\n", first, first)
}
输出结果:
值类型:gjson.Result,值:{"first": "Tom", "last": "Anderson"}
值类型:string,值:{"first": "Tom", "last": "Anderson"}
值类型:string,值:
值类型:string,值:Tom
2、类型转换
| 方法 | 描述 |
| String() | 尝试将此值转换为string |
| Int() | 尝试将此值转换为int64 |
| Uint64() | 尝试将此值转换为uint64 |
| Float64() | 尝试将此值转换为float64 |
| Bool() | 尝试将此值转换为布尔型 |
3、键路径
键路径实际上是以.分隔的一系列键。gjson支持在键中包含通配符*和?,*匹配任意多个字符,?匹配单个字符,例如ca*可以匹配cat/cate/cake等以ca开头的键,ca?只能匹配cat/cap等以ca开头且后面只有一个字符的键。
数组使用键名 + . + 索引(索引从 0 开始)的方式读取元素,如果键pets对应的值是一个数组,那么pets.0读取数组的第一个元素,pets.1读取第二个元素。
数组长度使用**键名 + . + #**获取,例如pets.#返回数组pets的长度。
如果键名中出现.,那么需要使用\进行转义
例(使用上面的json字符串):
//设置json字符串
jsonStr := `{"name": {"first": "Tom", "last": "Anderson"}, "age": 37, "children": ["Sara", "Alex", "Jack"], "friends": [{"first": "James", "last": "Murphy"}, {"first": "Roger", "last": "Craig"}]}` //获取数据
fmt.Println("name.last:", gjson.Get(jsonStr, "name.last"))
fmt.Println("age:", gjson.Get(jsonStr, "age"))
fmt.Println("children:", gjson.Get(jsonStr, "children"))
fmt.Println("children.#:", gjson.Get(jsonStr, "children.#")) //返回数组的长度
fmt.Println("children.1:", gjson.Get(jsonStr, "children.1")) //读取数组下标为1的元素值(注意下标从0开始)
fmt.Println("child*.2:", gjson.Get(jsonStr, "child*.2")) //*匹配,再读取下标为2的元素值
fmt.Println("c?ildren.0:", gjson.Get(jsonStr, "c?ildren.0")) //也是匹配,不过只匹配一个字符,再读取下标为0的元素值
fmt.Println("friends.#.first:", gjson.Get(jsonStr, "friends.#.first")) //如果数组#后还有内容,则以后面的路径读取数组中的每个元素,返回一个新的数组。所以该查询返回的数组所有friends的first字段组成
fmt.Println("friends.1.first:", gjson.Get(jsonStr, "friends.1.first")) //读取friends第 2 个元素的last字段
输出结果:
name.last: Anderson
age: 37
children: ["Sara", "Alex", "Jack"]
children.#: 3
children.1: Alex
child*.2: Jack
c?ildren.0: Sara
friends.#.first: ["James","Roger"]
friends.1.first: Roger
3.1 数组
对于数组,gjson还支持按条件查询元素,#(条件)返回第一个满足条件的元素,#(条件)#返回所有满足条件的元素。括号内的条件可以有==、!=、<、<=、>、>=,还有简单的模式匹配%(符合某个模式),!%(不符合某个模式):
package main import (
"fmt"
"github.com/tidwall/gjson"
) func main() {
//设置json字符串
var jsonStr = `{"name": {"first": "Tom", "last": "Anderson"}, "age": 37, "children": ["Sara", "Alex", "Jack"], "fav.movie": "Dear Hunter", "friends": [{"first": "Dale", "last": "Murphy", "age": 44, "nets": ["ig", "fb", "tw"]}, {"first": "Roger", "last": "Craig", "age": 68, "nets": ["fb", "tw"]}, {"first": "Jane", "last": "Murphy", "age": 47, "nets": ["ig", "tw"]}]}`
//{
// "name":{"first":"Tom", "last": "Anderson"},
// "age": 37,
// "children": ["Sara", "Alex", "Jack"],
//"fav.movie": "Dear Hunter",
//"friends": [
// {"first": "Dale", "last":"Murphy", "age": 44, "nets": ["ig", "fb", "tw"]},
// {"first": "Roger", "last": "Craig", "age": 68, "nets": ["fb", "tw"]},
// {"first": "Jane", "last": "Murphy", "age": 47, "nets": ["ig", "tw"]}
// ]
//} //输出数据
fmt.Println(gjson.Get(jsonStr, `friends.#(last="Murphy").first`)) //返回数组friends中第一个last为Murphy的元素;.first表示取出该元素的first字段返回
fmt.Println(gjson.Get(jsonStr, `friends.#(last="Murphy")#.first`)) //返回数组friends中所有的last为Murphy的元素,然后读取它们的first字段放在一个数组中返回
fmt.Println(gjson.Get(jsonStr, "friends.#(age>37)#.last")) //返回数组friends中所有年龄大于 45 的元素,然后读取它们的last字段返回
fmt.Println(gjson.Get(jsonStr, `friends.#(first%"D*").last`)) //返回数组friends中第一个first字段满足模式D*的元素,取出其last字段返回
fmt.Println(gjson.Get(jsonStr, `friends.#(first!%"D*").last`)) //返回数组friends中第一个first字段**不**满足模式D*的元素,读取其last`字段返回
fmt.Println(gjson.Get(jsonStr, `friends.#(nets.#(=="fb"))#.first`)) //这是个嵌套条件,friends.#(nets.#(=="fb"))#返回数组friends的元素的nets字段中有fb的所有元素,然后取出first字段返回
}
输出结果:
Dale
["Dale","Jane"]
["Murphy","Craig","Murphy"]
Murphy
Craig
["Dale","Roger"]
4、 一次获取多个值
调用gjson.Get()一次只能读取一个值,多次调用又比较麻烦,gjson提供了GetMany()可以一次读取多个值,返回一个数组[]gjson.Result。
//获取数据,使用3里面的字符串
results := gjson.GetMany(jsonStr, "name", "age", "children.#", "friends.#.first")
for _, result := range results {
fmt.Println(result)
}
//输出结果
// {"first": "Tom", "last": "Anderson"}
// 37
// 3
// ["Dale","Roger","Jane"]
5、校验json
调用gjson.Get()时,gjson假设我们传入的 JSON 串是合法的。如果 JSON 非法也不会panic,这时会返回不确定的结果
gjson.Valid()检测 JSON 串是否合法:
if !gjson.Valid(json) {
fmt.Println("error")
} else {
fmt.Println("ok")
}
6、遍历
gjson.Get()方法返回一个gjson.Result类型的对象,json.Result提供了ForEach()方法用于遍历。该方法接受一个类型为func (key, value gjson.Result) bool的回调函数。遍历对象时key和value分别为对象的键和值;遍历数组时,value为数组元素,key为空(不是索引)。回调返回false时,遍历停止。
results := gjson.Get(jsonStr, "friends") //获取初始数据,3里面的字符串
results.ForEach(func(k, result gjson.Result) bool { //遍历,一般不需要k,可用_忽略
first := gjson.Get(result.Raw, "first") //可以再次根据键路径获取更为详细的数据
fmt.Printf("k:%v,v:%v\n", k, results) //输出键值
fmt.Println(first) //输出friends数组里的first元素值
return true
})
输出结果:
k:0,v:[{"first": "Dale", "last": "Murphy", "age": 44, "nets": ["ig", "fb", "tw"]}, {"first": "Roger", "last": "Craig", "age": 68, "nets": ["fb", "tw"]}, {"first": "Jane", "last": "Murphy", "age": 47, "nets": ["ig", "tw"]}]
Dale
k:1,v:[{"first": "Dale", "last": "Murphy", "age": 44, "nets": ["ig", "fb", "tw"]}, {"first": "Roger", "last": "Craig", "age": 68, "nets": ["fb", "tw"]}, {"first": "Jane", "last": "Murphy", "age": 47, "nets": ["ig", "tw"]}]
Roger
k:2,v:[{"first": "Dale", "last": "Murphy", "age": 44, "nets": ["ig", "fb", "tw"]}, {"first": "Roger", "last": "Craig", "age": 68, "nets": ["fb", "tw"]}, {"first": "Jane", "last": "Murphy", "age": 47, "nets": ["ig", "tw"]}]
Jane
PS D:\Users\W9010315\GolandProjects\test> go run .\gjson.go
k:0,v:[{"first": "Dale", "last": "Murphy", "age": 44, "nets": ["ig", "fb", "tw"]}, {"first": "Roger", "last": "Craig", "age": 68, "nets": ["fb", "tw"]}, {"first": "Jane", "last": "Murphy", "age": 47, "nets": ["ig", "tw"]}]
Dale
k:1,v:[{"first": "Dale", "last": "Murphy", "age": 44, "nets": ["ig", "fb", "tw"]}, {"first": "Roger", "last": "Craig", "age": 68, "nets": ["fb", "tw"]}, {"first": "Jane", "last": "Murphy", "age": 47, "nets": ["ig", "tw"]}]
Roger
k:2,v:[{"first": "Dale", "last": "Murphy", "age": 44, "nets": ["ig", "fb", "tw"]}, {"first": "Roger", "last": "Craig", "age": 68, "nets": ["fb", "tw"]}, {"first": "Jane", "last": "Murphy", "age": 47, "nets": ["ig", "tw"]}]
Jane
本文参考:https://darjun.github.io/2020/03/22/godailylib/gjson/
7、补充
7.1 Exists()
该函数用于检查指定的键是否存在于json对象中,存在则返回true,否则返回false。可用来检测填写的键路径是否正确
// 解析 JSON 数据
if gjson.Get(jsonStr, "name").Exists() {
fmt.Println("name 键存在")
} else {
fmt.Println("name 键不存在")
}
Go--gjson的更多相关文章
- 解决GJson 获取web api数据出现Not a JsonObject问题
服务器端web api服务采用asp.net web api编写,对请求的数据序列化成Json格式的字符串进行传递. 客户端采用Java进行接收处理,处理采用GJson进行解析,出现Not a Jso ...
- Tomjson - 一个"短小精悍"的 json 解析库
Tomjson,一个"短小精悍"的 json 解析库,tomjson使用Java语言编写,主要作用是把Java对象(JavaBean)序列化为json格式字符串,将json格式字符 ...
- JavaSE18章_JSON解析详解
一.JSON简介 JSON(JavaScript Object Notation),是一种轻量级的数据交换格式.JSON是存储和交换文本信息的,语法类似 XML.易于人阅读和编写,同时也易于机器解析和 ...
- android中Json的一些应用
JSON(JavaScript Object Notation) :一种轻量级的数据交换格式,基于JavaScript的一个子集. JSON采用完全独立于语言的文本格式,使JSON成为理想的数据交换语 ...
- android json解析详细介绍之gson
废话不多说,什么json是轻量级数据交换标准:自己百度去深入了解:这里有三种json解析工具.本人只用过其中两种: 1.Google Json利器之Gson 评价:简单,方便. 2.阿里巴巴 ...
- js,jquery转json的几种方法
一.原生js转json, eval()方法,不需要引入外部插件; //由JSON字符串转换为JSON对象 var obj = eval('(' + jsonStr + ')'); 或者 var obj ...
- 第一次使用Android Studio时你应该知道的一切配置(三):gradle项目构建
[声明] 欢迎转载,但请保留文章原始出处→_→ 生命壹号:http://www.cnblogs.com/smyhvae/ 文章来源:http://www.cnblogs.com/smyhvae/p/ ...
- 第一次使用Android Studio时你应该知道的一切配置(二):新建一个属于自己的工程并安装Genymotion模拟器
[声明] 欢迎转载,但请保留文章原始出处→_→ 生命壹号:http://www.cnblogs.com/smyhvae/ 文章来源:http://www.cnblogs.com/smyhvae/p/4 ...
- 【转】第一次使用Android Studio时你应该知道的一切配置(三):gradle项目构建
原文网址:http://www.cnblogs.com/smyhvae/p/4456420.html [声明] 欢迎转载,但请保留文章原始出处→_→ 生命壹号:http://www.cnblogs.c ...
- Tomjson - json 解析库
Tomjson - 一个"短小精悍"的 json 解析库 Tomjson,一个"短小精悍"的 json 解析库,tomjson使用Java语言编写,主要作用是把 ...
随机推荐
- 推荐一个Node.js多版本管理的可视化工具
关于Node.js的开发者来说,在开发机器上管理多个不同版本的Node.js是一个常见痛点.之前在开发者安全大全专栏中,提到过解决方法:使用nvm,如果对于nvm还不了解的话,可以前往了解. 对于TJ ...
- 算法训练 字符串的展开(isdigit与islower的发现)
在初赛普及组的"阅读程序写结果"的问题中,我们曾给出一个字符串展开的例子:如果在输入的字符串中,含有类似于"d-h"或者"4-8"的字串,我 ...
- go基础-函数
概述 在任何语言中函数都是极其重要的内容,业务功能都是由一个或多个函数组合完成.go语言是函数式编程语言,函数是一等公民,可以被传递.有函数类型,go语言有三种类型的函数,普通函数.匿名函数(Lamb ...
- Python 潮流周刊#28:两种线程池、四种优化程序的方法
你好,我是猫哥.这里每周分享优质的 Python.AI 及通用技术内容,大部分为英文.本周刊开源,欢迎投稿.另有电报频道作为副刊,补充发布更加丰富的资讯. 产品推荐 Walles.AI 是一款适用于所 ...
- 报错:bs4.FeatureNotFound: Couldn't find a tree builder with the features you requ
安装 pip3 install lxml 即可
- AntDesignBlazor示例——列表查询条件
本示例是AntDesign Blazor的入门示例,在学习的同时分享出来,以供新手参考. 示例代码仓库:https://gitee.com/known/AntDesignDemo 1. 学习目标 重构 ...
- kotlin+springboot入门级别教程,教你如何用kotlin和springboot搭建http
先打开idea,或者用springboot官网.阿里云那边都行 然后点击新建项目,spring Initializr,我们都知道,springboot是支持kotlin的,除非你是kotlin1.3之 ...
- jvm总结图解
浅析jvm 内存模型 https://www.cnblogs.com/lewis0077/p/5143268.html
- 开源 Serverless 框架 Laf 性能优化实践
介绍 Laf 是一个完全开源的 Serverless 框架,Laf 的 Node.js 运行时容器 (以下简称为 Runtime) 是 Laf 的函数执行环境,依托于 Express.js 框架.采用 ...
- 数字孪生技术结合GIS系统能在农业领域作出什么改变?
数字孪生技术和地理信息系统(GIS)是两个独立但高度互补的领域,它们的结合在农业领域具有巨大的潜力,可以带来巨大的改变.在这篇文章中,我们将讨论数字孪生技术和GIS系统如何协同作用,为农业带来创新和可 ...