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语言编写,主要作用是把 ...
随机推荐
- Android 使用 ContentProvider 简单操作数据库
ContentProvider 可以用来原生读写 Android 自带的数据库 SQLite. 使用 Studio 创建一个 ContentProvider, 名字叫 TestContentProvi ...
- LabVIEW基于机器视觉的实验室设备管理系统(2)
目录 功能计划 普通用户 欢迎登录 信息查询 返回退出程序 效果演示 在前期的准备完成之后呢,就要开始实现我们的程序啦,不过在编程之前,我们一定要计划好这个系统需要什么功能. 功能计划 既然我们做的是 ...
- GUI界面实现小学生口算题卡功能(一)| 简要了解GUI
上课没认真听,下课不好好写. 关于GUI,首先了解了一下什么是GUI: GUI(Graphical User Interface),图形用户界面.采用图形方式显示的计算机操作用户接口.与早期计算机使用 ...
- NodeJS下载安装
一.什么是NodeJS? NodeJS是一个开源,跨平台的JavaScript运行环境 二.NodeJS安装包下载 1.打开网址:Node.js (nodejs.org) 2.下载稳定版本 三.Nod ...
- 华企盾DSC防泄密:有一个文件申请解密失败,提示拒绝访问(密钥不对)
解决方法:将文件拖到加密进程中提示密钥不对,找到原来的数据库还原解密
- 华企盾DSC邮件服务器测试连接提示Server has closed the connection(端口不对)
解决方法:邮件服务器端口填错了,应该是smtp.126.com:s465,或者smtp.126.com:s587 其他邮箱同理.
- 【Python】【OpenCV】Cameo项目(一)实时显示摄像头帧
Cameo项目介绍: 1.实时捕获并显示摄像头帧. 2.具备截图.保存视频和退出三个功能键. 要求存在文件:manager.py 和 cameo.py 一.manager.py 两个类:Capture ...
- JavaScript String对象及方法总结
String 对象创建方法: new String() var txt1 = new String("string"); var txt2 = "string" ...
- 【JMM内存模型-4】JMM内存模型之CPU缓存策略-jmmcpu4
title: [JMM内存模型-4]JMM内存模型之CPU缓存策略 date: 2021-11-17 13:27:48.139 updated: 2021-12-26 17:43:10.442 url ...
- elastic优化
通过定义keyword 的 "null_value" :"NULL",使得搜索是不用单独使用exists查询.统一用terms查询就能查询到想要的结果 利用co ...