当前位置: 首页 > news >正文

深入解析 JSONPath:从入门到精通

码到三十五 : 个人主页

在数据处理和交换领域,JSON已经成为了一种广泛使用的数据格式,
如何有效地查询和操作这些数据也变得越来越重要。在这种情况下,JSONPath 应运而生,成为了一种在JSON数据中定位和提取信息的强大工具。

目录

    • 一、什么是 JSONPath
    • 二、JSONPath 基本语法
    • 三、JSONPath 高级特性
    • 四、JSONPath 应用场景
    • 五、JSONPath的使用
    • 结语

一、什么是 JSONPath

JSONPath 是一种在JSON数据中查询信息的表达式语言,它允许用户通过一种简洁明了的语法来定位和提取JSON对象中的特定数据。与XML的XPath类似,JSONPath 提供了一种灵活且强大的方式来查询JSON结构中的数据。

二、JSONPath 基本语法

JSONPath 的语法相对简单,但功能却非常强大。以下是一些基本的语法规则:

  1. $:表示JSON数据的根对象。
  2. .[]:用于访问对象的属性或数组的元素。例如,$.name 或 $[‘name’] 都可以访问根对象中的 ‘name’ 属性。
  3. :表示递归下降,用于查找所有级别的属性。
  4. ?():应用一个过滤表达式来过滤数组中的元素。例如,$?(@.age>18) 将选择所有年龄大于18的对象。
  5. []:在属性名或数组索引位置使用,表示选择所有元素。例如,$.students[*].name 将选择所有学生的名字。
  6. -101n:用作数组索引时,表示从最后一个元素开始计数。例如,$.students[-1].name 将选择最后一个学生的名字。

在这里插入图片描述

三、JSONPath 高级特性

除了基本语法之外,JSONPath 还提供了一些高级特性,使得数据查询更加灵活和强大。

  1. 通配符与切片:你可以使用 * 通配符来选择所有属性,或者使用切片语法(如 [start:end:step])来选择数组中的特定元素范围。
  2. 函数:JSONPath 支持一些内置函数,如 length()(获取数组或字符串长度)、keys()(获取对象所有键)等,这些函数可以在查询中进行更复杂的操作。
  3. 条件表达式:通过结合使用 ?() 和逻辑操作符(如 &&||),你可以构建复杂的条件表达式来过滤数据。

四、JSONPath 应用场景

JSONPath 在多个领域都有广泛的应用,包括但不限于:

  1. 数据验证:通过 JSONPath 表达式,你可以轻松地验证 JSON 数据的结构和内容是否符合预期。
  2. 数据提取与转换:在处理大量 JSON 数据时,JSONPath 可以帮助你快速定位和提取所需信息,或者将数据转换为其他格式。
  3. 自动化测试:在自动化测试中,你可以使用 JSONPath 来验证 API 响应中的数据是否符合预期。
  4. 日志分析:对于包含 JSON 格式的日志文件,JSONPath 可以帮助你快速提取和分析关键信息。

五、JSONPath的使用

以下是一些JSONPath的使用,展示了如何使用JSONPath表达式从JSON数据中提取信息。

假设我们有以下JSON数据:

{"store": {"book": [{"title": "Sword of Honour","price": 12.99},{"title": "Moby Dick","price": 8.99},{"title": "The Lord of the Rings","price": 22.99}],"bicycle": {"color": "red","price": 19.95}},"expensive": 10
}

首先,需要将JsonPath库添加到项目中。如果你使用Maven,可以在pom.xml文件中添加以下依赖:

<dependency><groupId>com.jayway.jsonpath</groupId><artifactId>json-path</artifactId><version>2.7.0</version> <!-- 请检查是否有更新的版本 -->
</dependency>

接下来是Java代码:

import com.jayway.jsonpath.JsonPath;public class JsonPathExample {public static void main(String[] args) {String json = "{\n" +"    \"store\": {\n" +"        \"book\": [\n" +"            {\n" +"                \"title\": \"Sword of Honour\",\n" +"                \"price\": 12.99\n" +"            },\n" +"            {\n" +"                \"title\": \"Moby Dick\",\n" +"                \"price\": 8.99\n" +"            },\n" +"            {\n" +"                \"title\": \"The Lord of the Rings\",\n" +"                \"price\": 22.99\n" +"            }\n" +"        ],\n" +"        \"bicycle\": {\n" +"            \"color\": \"red\",\n" +"            \"price\": 19.95\n" +"        }\n" +"    },\n" +"    \"expensive\": 10\n" +"}\n";// 提取所有的书名String bookTitlesPath = "$.store.book[*].title";Object bookTitles = JsonPath.read(json, bookTitlesPath);System.out.println("Book Titles: " + bookTitles);// 提取第一本书的价格String firstBookPricePath = "$.store.book[0].price";Object firstBookPrice = JsonPath.read(json, firstBookPricePath);System.out.println("First Book Price: " + firstBookPrice);// 提取价格大于10的书名String expensiveBookTitlesPath = "$.store.book[?(@.price > 10)].title";Object expensiveBookTitles = JsonPath.read(json, expensiveBookTitlesPath);System.out.println("Expensive Book Titles: " + expensiveBookTitles);}
}

首先定义了一个JSON字符串json,然后使用JsonPath.read方法来执行JSONPath查询。分别查询了所有的书名、第一本书的价格以及价格大于10的书名,并将结果打印出来。

下面是使用上述JSON数据的更多JSONPath用法:

  1. 提取bicycle的颜色

    JSONPath 表达式: $.store.bicycle.color

    String bicycleColorPath = "$.store.bicycle.color";
    Object bicycleColor = JsonPath.read(json, bicycleColorPath);
    System.out.println("Bicycle Color: " + bicycleColor);
    
  2. 提取不是"Sword of Honour"的所有书名

    为了提取不等于"Sword of Honour"的书名,我们可以使用!=操作符。但请注意,不是所有的JSONPath实现都支持这种比较操作。如果你的实现不支持,你可能需要在应用层面进行过滤。

    假设我们的JSONPath库支持这种比较,表达式可能类似于:

    JSONPath 表达式: $.store.book[?(@.title != 'Sword of Honour')].title

    String notSwordOfHonourPath = "$.store.book[?(@.title != 'Sword of Honour')].title";
    Object notSwordOfHonourTitles = JsonPath.read(json, notSwordOfHonourPath);
    System.out.println("Book Titles Not 'Sword of Honour': " + notSwordOfHonourTitles);
    
  3. 提取最贵的书的价格

    为了获取最贵的书的价格,我们可以先获取所有书的价格,然后在应用层面找到最大值。但如果JSONPath实现支持,我们也可以直接在表达式中使用max()函数。

    JSONPath 表达式(如果支持): $.store.book[*].price.max()

    在标准的JsonPath中并不直接支持这样的聚合函数,因此你可能需要在Java代码中处理这个问题:

    String allPricesPath = "$.store.book[*].price";
    List<Double> allPrices = JsonPath.read(json, allPricesPath);
    double maxPrice = Collections.max(allPrices);
    System.out.println("Maximum Book Price: " + maxPrice);
    
  4. 检查是否有价格超过20的书

    JSONPath 本身不直接支持返回一个布尔值来表示是否存在满足条件的元素,但你可以在获取结果后判断结果集合是否为空。

    JSONPath 表达式: $.store.book[?(@.price > 20)]

    String expensiveBooksPath = "$.store.book[?(@.price > 20)]";
    Object expensiveBooks = JsonPath.read(json, expensiveBooksPath);
    boolean hasExpensiveBooks = ((List<?>) expensiveBooks).size() > 0;
    System.out.println("Has books priced over 20: " + hasExpensiveBooks);
    
  5. 获取bicycle的价格,并判断其是否大于15

    首先提取bicycle的价格,然后在Java代码中做比较。

    JSONPath 表达式: $.store.bicycle.price

    String bicyclePricePath = "$.store.bicycle.price";
    Object bicyclePriceObj = JsonPath.read(json, bicyclePricePath);
    double bicyclePrice = Double.parseDouble(bicyclePriceObj.toString());
    boolean isBicyclePriceGreaterThan15 = bicyclePrice > 15;
    System.out.println("Is bicycle price greater than 15? " + isBicyclePriceGreaterThan15);
    

由于JSONPath的具体实现可能有所不同,某些高级功能(如过滤、聚合等)可能不在所有实现中都可用。如果你使用的JsonPath库不支持这些功能,你可能需要在Java代码中实现相应的逻辑。

结语

JSONPath 作为一种强大的 JSON 数据查询语言,为我们提供了便捷的数据定位和提取方式。通过深入学习和实践 JSONPath,我们能够更好地处理和利用 JSON 数据,为应用带来更大的便捷。


听说...关注下面公众号的人都变牛了,纯技术,纯干货 !

相关文章:

  • C#记录日志
  • Shell脚本的分支语句,循环语句
  • Vue 3 教程:核心知识
  • 29-ESP32-S3-WIFI_Driver-00 STA模式扫描全部 AP
  • 出现 java: Annotation processing is not supported for module cycles 解决方法
  • 2024年蓝桥杯Web开发【大赛大纲】15届
  • 基于51单片机的温控风扇的设计–仿真设计
  • WebRTC | 网络传输协议 RTP 和 RTCP
  • JavaScript 中遍历数组的多种方法
  • 2.10 mysql设置远程访问权限
  • 【第7章】SpringBoot整合Mybatis-Plus
  • 2024年QMT智能量化交易全解读:一文带你深入了解什么是QMT
  • 长安链使用Golang编写智能合约教程(二)
  • 深度解析搜索引擎广告(SEM)与社交媒体广告(SMM):NetFarmer助力企业数字化出海
  • 【QT八股文】系列之篇章3 | QT的多线程以及QThread与QObject
  • 【Leetcode】101. 对称二叉树
  • Bootstrap JS插件Alert源码分析
  • Electron入门介绍
  • Fundebug计费标准解释:事件数是如何定义的?
  • JavaScript 基础知识 - 入门篇(一)
  • Linux链接文件
  • Python进阶细节
  • SpiderData 2019年2月13日 DApp数据排行榜
  • Swoft 源码剖析 - 代码自动更新机制
  • vue-cli在webpack的配置文件探究
  • 发布国内首个无服务器容器服务,运维效率从未如此高效
  • 搞机器学习要哪些技能
  • 给第三方使用接口的 URL 签名实现
  • 基于游标的分页接口实现
  • 聊聊flink的TableFactory
  • 前端技术周刊 2019-02-11 Serverless
  • 日剧·日综资源集合(建议收藏)
  • 深度学习中的信息论知识详解
  • 深入 Nginx 之配置篇
  • 在weex里面使用chart图表
  • 摩拜创始人胡玮炜也彻底离开了,共享单车行业还有未来吗? ...
  • ​七周四次课(5月9日)iptables filter表案例、iptables nat表应用
  • ​人工智能书单(数学基础篇)
  • ​软考-高级-系统架构设计师教程(清华第2版)【第15章 面向服务架构设计理论与实践(P527~554)-思维导图】​
  • # 消息中间件 RocketMQ 高级功能和源码分析(七)
  • ### Error querying database. Cause: com.mysql.jdbc.exceptions.jdbc4.CommunicationsException
  • #define 用法
  • #HarmonyOS:Web组件的使用
  • $var=htmlencode(“‘);alert(‘2“); 的个人理解
  • (10)ATF MMU转换表
  • (JS基础)String 类型
  • (Matalb时序预测)WOA-BP鲸鱼算法优化BP神经网络的多维时序回归预测
  • (NSDate) 时间 (time )比较
  • (Oracle)SQL优化基础(三):看懂执行计划顺序
  • (solr系列:一)使用tomcat部署solr服务
  • (附源码)ssm考试题库管理系统 毕业设计 069043
  • (附源码)流浪动物保护平台的设计与实现 毕业设计 161154
  • (三分钟了解debug)SLAM研究方向-Debug总结
  • (一)Linux+Windows下安装ffmpeg
  • (源码版)2024美国大学生数学建模E题财产保险的可持续模型详解思路+具体代码季节性时序预测SARIMA天气预测建模