Spring Boot中高效提取嵌套JSON数据的策略

Spring Boot中高效提取嵌套JSON数据的策略

本教程详细探讨了在spring boot应用中如何高效地从复杂嵌套json结构中提取特定数据。我们将重点介绍jackson库的两种核心方法:jackson streaming api,适用于处理大型或结构动态的json,以及jackson data binding,适用于将json映射到预定义java对象。文章将提供详细的代码示例和实现步骤,并讨论如何进一步过滤提取出的数据,例如按类别筛选。

在Spring Boot应用程序中处理来自API调用的JSON数据是常见的任务,尤其当JSON结构包含多层嵌套时,如何高效且优雅地提取所需信息成为一个关问题。本文将深入探讨使用Jackson库来解决这一挑战的两种主要方法:Jackson Streaming API和Jackson Data Binding,并提供具体的代码示例。

1. 准备工作

在Spring Boot项目中,Jackson是默认的JSON处理库,因此通常无需额外添加依赖。但如果需要确保版本或解决特定问题,可以检查 pom.xml 中是否包含以下依赖:

    com.fasterxml.jackson.core    jackson-databind

假设我们有以下JSON数据,其中包含嵌套的 footwearList 和 clothingList:

[    {        "id": 1,        "footwearList": [            {                "id": 1,                "name": "sandals",                "category": "men"            },            {                "id": 3,                "name": "sandals",                "category": "women"            }        ],        "clothingList": [            {                "id": 1,                "name": "t-shirt",                "category": "men"            },            {                "id": 3,                "name": "tshirt",                "category": "women"            }        ]    },    {        "id": 2,        "footwearList": [            {                "id": 2,                "name": "shoes",                "category": "men"            },            {                "id": 4,                "name": "shoes",                "category": "women"            }        ],        "clothingList": [            {                "id": 2,                "name": "shirt",                "category": "men"            },            {                "id": 4,                "name": "shirt",                "category": "women"            }        ]    }]

我们的目标是从这个JSON中提取 footwearList 和 clothingList 中的 Item 对象,并可能进一步按 category 进行筛选。

2. 使用Jackson Streaming API

当JSON结构非常庞大、键名动态不固定,或者我们不希望将整个JSON完全映射为Java对象时,Jackson Streaming API提供了一种高效的迭代方式。它允许我们遍历JSON树,按需提取数据。

首先,定义一个 Item 类来表示 footwearList 和 clothingList 中的元素:

public class Item {    private int id;    private String name;    private String category;    // 无参构造函数    public Item() {}    // 带参构造函数    public Item(int id, String name, String category) {        this.id = id;        this.name = name;        this.category = category;    }    // Getters and Setters    public int getId() { return id; }    public void setId(int id) { this.id = id; }    public String getName() { return name; }    public void setName(String name) { this.name = name; }    public String getCategory() { return category; }    public void setCategory(String category) { this.category = category; }    @Override    public String toString() {        return "Item{id=" + id + ", name='" + name + "', category='" + category + "'}";    }}

接下来,我们创建一个辅助方法将 JsonNode 转换为 Item 对象:

import com.fasterxml.jackson.databind.JsonNode;import com.fasterxml.jackson.databind.ObjectMapper;public class JsonExtractor {    private static final ObjectMapper MAPPER = new ObjectMapper();    // 辅助方法:将JsonNode转换为Item对象    public static Item nodeToItem(JsonNode node) {        try {            return MAPPER.treeToValue(node, Item.class);        } catch (com.fasterxml.jackson.core.JsonProcessingException e) {            System.err.println("Error converting JsonNode to Item: " + e.getMessage());            return null;        }    }    // ... 其他方法}

现在,我们可以编写逻辑来遍历JSON并提取嵌套列表中的 Item:

import com.fasterxml.jackson.databind.JsonNode;import com.fasterxml.jackson.databind.ObjectMapper;import java.util.List;import java.util.regex.Pattern;import java.util.function.Predicate;import java.util.stream.StreamSupport;import java.util.Collections; // 导入Collectionspublic class JsonExtractor {    private static final ObjectMapper MAPPER = new ObjectMapper();    // 定义一个正则表达式谓词,用于匹配包含"footwear"或"clothing"的字段名    public static final Predicate TARGET_LIST_PREDICATE = Pattern.compile("footwear|clothing").asPredicate();    // 辅助方法:将JsonNode转换为Item对象 (同上)    public static Item nodeToItem(JsonNode node) {        try {            return MAPPER.treeToValue(node, Item.class);        } catch (com.fasterxml.jackson.core.JsonProcessingException e) {            System.err.println("Error converting JsonNode to Item: " + e.getMessage());            return null;        }    }    public List extractNestedItems(String json) {        try {            JsonNode rootNode = MAPPER.readTree(json);            if (!rootNode.isArray()) {                System.err.println("Root JSON node is not an array.");                return Collections.emptyList();            }            return StreamSupport.stream(rootNode.spliterator(), false) // 遍历根数组的每个对象                .mapMulti((node, consumer) -> { // 对于每个对象                    node.fields().forEachRemaining(entry -> { // 遍历其所有字段                        if (TARGET_LIST_PREDICATE.test(entry.getKey()) && entry.getValue().isArray()) {                            // 如果字段名匹配且其值为数组,则将数组中的每个元素传递给consumer                            entry.getValue().forEach(consumer);                        }                    });                })                .map(JsonExtractor::nodeToItem) // 将每个JsonNode转换为Item对象                .toList(); // 收集结果        } catch (com.fasterxml.jackson.core.JsonProcessingException e) {            System.err.println("Error parsing JSON: " + e.getMessage());            return Collections.emptyList();        }    }    // 示例用法    public static void main(String[] args) {        String json = "[n" +                "    {n" +                "        "id": 1,n" +                "        "footwearList": [n" +                "            { "id": 1, "name": "sandals", "category": "men" },n" +                "            { "id": 3, "name": "sandals", "category": "women" }n" +                "        ],n" +                "        "clothingList": [n" +                "            { "id": 1, "name": "t-shirt", "category": "men" },n" +                "            { "id": 3, "name": "tshirt", "category": "women" }n" +                "        ]n" +                "    },n" +                "    {n" +                "        "id": 2,n" +                "        "footwearList": [n" +                "            { "id": 2, "name": "shoes", "category": "men" },n" +                "            { "id": 4, "name": "shoes", "category": "women" }n" +                "        ],n" +                "        "clothingList": [n" +                "            { "id": 2, "name": "shirt", "category": "men" },n" +                "            { "id": 4, "name": "shirt", "category": "women" }n" +                "        ]n" +                "    }n" +                "]";        JsonExtractor extractor = new JsonExtractor();        List allItems = extractor.extractNestedItems(json);        System.out.println("所有提取的Item:");        allItems.forEach(System.out::println);        // 进一步按类别过滤        List menItems = allItems.stream()                                      .filter(item -> "men".equals(item.getCategory()))                                      .toList();        System.out.println("n按类别'men'过滤的Item:");        menItems.forEach(System.out::println);    }}

输出示例:

所有提取的Item:Item{id=1, name='sandals', category='men'}Item{id=3, name='sandals', category='women'}Item{id=1, name='t-shirt', category='men'}Item{id=3, name='tshirt', category='women'}Item{id=2, name='shoes', category='men'}Item{id=4, name='shoes', category='women'}Item{id=2, name='shirt', category='men'}Item{id=4, name='shirt', category='women'}按类别'men'过滤的Item:Item{id=1, name='sandals', category='men'}Item{id=1, name='t-shirt', category='men'}Item{id=2, name='shoes', category='men'}Item{id=2, name='shirt', category='men'}

这种方法非常灵活,即使JSON的顶层结构或嵌套列表的键名发生变化,只要我们调整 TARGET_LIST_PREDICATE,代码仍然可以工作。

3. 使用Jackson Data Binding

对于结构相对固定且已知其模式的JSON数据,Jackson Data Binding是更推荐和简洁的方法。它允许我们将JSON直接映射到Java对象(POJO),极大地简化了数据访问

首先,定义一个 Store 类来表示JSON中的顶层对象。由于 footwearList 和 clothingList 的键是动态的,我们可以使用 Map<String, List> 结合 @JsonAnySetter 和 @JsonAnyGetter 来处理。

import com.fasterxml.jackson.annotation.JsonAnyGetter;import com.fasterxml.jackson.annotation.JsonAnySetter;import java.util.HashMap;import java.util.List;import java.util.Map;public class Store {    private int id;    private Map<String, List> items = new HashMap(); // 用于存储动态的列表    // 无参构造函数    public Store() {}    // Getters and Setters for id    public int getId() { return id; }    public void setId(int id) { this.id = id; }    // @JsonAnySetter 用于反序列化时捕获未映射的键值对    @JsonAnySetter    public void readStore(String key, List value) {        this.items.put(key, value);    }    // @JsonAnyGetter 用于序列化时将Map中的内容作为顶级字段输出    // 注意:这里的实现是为了演示,实际序列化时可能需要更精细的控制    @JsonAnyGetter    public Map<String, List> getItems() {        return items;    }    @Override    public String toString() {        return "Store{id=" + id + ", items=" + items + '}';    }}

现在,我们可以使用 ObjectMapper 将整个JSON字符串反序列化为 List:

import com.fasterxml.jackson.core.type.TypeReference;import com.fasterxml.jackson.databind.ObjectMapper;import java.util.Collections;import java.util.List;import java.util.stream.Collectors;public class JsonDataBinder {    private static final ObjectMapper MAPPER = new ObjectMapper();    public List parseJsonToStores(String json) {        try {            return MAPPER.readValue(json, new TypeReference<List>() {});        } catch (com.fasterxml.jackson.core.JsonProcessingException e) {            System.err.println("Error parsing JSON to Store list: " + e.getMessage());            return Collections.emptyList();        }    }    public static void main(String[] args) {        String json = "[n" +                "    {n" +                "        "id": 1,n" +                "        "footwearList": [n" +                "            { "id": 1, "name": "sandals", "category": "men" },n" +                "            { "id": 3, "name": "sandals", "category": "women" }n" +                "        ],n" +                "        "clothingList": [n" +                "            { "id": 1, "name": "t-shirt", "category": "men" },n" +                "            { "id": 3, "name": "tshirt", "category": "women" }n" +                "        ]n" +                "    },n" +                "    {n" +                "        "id": 2,n" +                "        "footwearList": [n" +                "            { "id": 2, "name": "shoes", "category": "men" },n" +                "            { "id": 4, "name": "shoes", "category": "women" }n" +                "        ],n" +                "        "clothingList": [n" +                "            { "id": 2, "name": "shirt", "category": "men" },n" +                "            { "id": 4, "name": "shirt", "category": "women" }n" +                "        ]n" +                "    }n" +                "]";        JsonDataBinder binder = new JsonDataBinder();        List stores = binder.parseJsonToStores(json);        System.out.println("所有解析的Store对象:");        stores.forEach(System.out::println);        // 提取所有鞋类商品        List allFootwear = stores.stream()                                       .flatMap(store -> store.getItems().getOrDefault("footwearList", Collections.emptyList()).stream())                                       .collect(Collectors.toList());        System.out.println("n所有鞋类商品:");        allFootwear.forEach(System.out::println);        // 提取所有男士服装        List menClothing = stores.stream()                                       .flatMap(store -> store.getItems().getOrDefault("clothingList", Collections.emptyList()).stream())                                       .filter(item -> "men".equals(item.getCategory()))                                       .collect(Collectors.toList());        System.out.println("n所有男士服装:");        menClothing.forEach(System.out::println);    }}

输出示例:

所有解析的Store对象:Store{id=1, items={footwearList=[Item{id=1, name='sandals', category='men'}, Item{id=3, name='sandals', category='women'}], clothingList=[Item{id=1, name='t-shirt', category='men'}, Item{id=3, name='tshirt', category='women'}]}}Store{id=2, items={footwearList=[Item{id=2, name='shoes', category='men'}, Item{id=4, name='shoes', category='women'}], clothingList=[Item{id=2, name='shirt', category='men'}, Item{id=4, name='shirt', category='women'}]}}所有鞋类商品:Item{id=1, name='sandals', category='men'}Item{id=3, name='sandals', category='women'}Item{id=2, name='shoes', category='men'}Item{id=4, name='shoes', category='women'}所有男士服装:Item{id=1, name='t-shirt', category='men'}Item{id=2, name='shirt', category='men'}

Data Binding方法将整个JSON结构映射为强类型Java对象,使得后续的数据访问和操作(如使用Java Stream API进行过滤)变得非常直观和类型安全。

4. 注意事项与总结

选择合适的解析策略:Jackson Streaming API: 适用于JSON结构不固定、键名动态、数据量非常大(避免一次性加载所有数据到内存)的场景。它提供了更细粒度的控制,但代码相对复杂。Jackson Data Binding: 适用于JSON结构相对稳定且可以预先定义POJO的场景。它提供了更高的开发效率和代码可读性,是大多数Spring Boot应用的首选。错误处理: 在实际应用中,始终要包含 try-catch 块来处理 JsonProcessingException(或其子类 IOException),以优雅地处理JSON解析过程中可能出现的错误。性能: 对于中小型JSON,Data Binding的性能通常足够好。对于超大型JSON,Streaming API可能提供更好的内存效率。JSONPath: 虽然原问题中提到了 JsonPath,它是一个强大的JSON查询语言。但在Spring Boot的Jackson生态系统中,通常可以直接利用Jackson本身的API来实现大部分需求,而无需引入额外的 JsonPath 依赖,从而保持项目依赖的简洁性。

通过掌握Jackson Streaming API和Data Binding这两种方法,开发者可以根据具体需求,在Spring Boot中灵活高效地处理各种复杂嵌套的JSON数据。在大多数情况下,Data Binding结合 @JsonAnySetter 等注解,能够以最简洁的方式解决嵌套和动态键的问题。

以上就是Spring Boot中高效提取嵌套JSON数据的策略的详细内容,更多请关注创想鸟其它相关文章!

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 chuangxiangniao@163.com 举报,一经查实,本站将立刻删除。
发布者:程序猿,转转请注明出处:https://www.chuangxiangniao.com/p/195487.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2025年11月1日 18:10:21
下一篇 2025年11月1日 18:13:31

相关推荐

  • 云闪付怎么快速赚取积点_云闪付积点快速获取方法

    通过微信小程序用云闪付支付可日赚692积点;62VIP会员消费满10元返积点,月上限3000;转账超1000元得2积点,还款超100元得10积点,每月各限3笔;扫本人收款码支付5元以上每笔得10积点,日限3笔;改定位至杭州领“浙里有优惠”活动卡可得2025积点。 如果您在使用云闪付时希望快速积累积点…

    2025年12月6日 软件教程
    400
  • AO3镜像站备用镜像网址_AO3镜像站快速访问官网

    AO3镜像站备用网址包括ao3mirror.com和xiaozhan.icu,当主站archiveofourown.org无法访问时可切换使用,二者均同步更新内容并支持多语言检索与离线下载功能。 AO3镜像站备用镜像网址在哪里?这是不少网友都关注的,接下来由PHP小编为大家带来AO3镜像站快速访问官…

    2025年12月6日 软件教程
    100
  • 天猫app淘金币抵扣怎么使用

    在天猫app购物时,淘金币是一项能够帮助你节省开支的实用功能。掌握淘金币的抵扣使用方法,能让你以更实惠的价格买到心仪商品。 当你选好商品并准备下单时,记得查看商品页面是否支持淘金币抵扣。如果该商品支持此项功能,在提交订单的页面会明确显示相关提示。你会看到淘金币的具体抵扣比例——通常情况下,淘金币可按…

    2025年12月6日 软件教程
    500
  • 怎样用免费工具美化PPT_免费美化PPT的实用方法分享

    利用KIMI智能助手可免费将PPT美化为科技感风格,但需核对文字准确性;2. 天工AI擅长优化内容结构,提升逻辑性,适合高质量内容需求;3. SlidesAI支持语音输入与自动排版,操作便捷,利于紧急场景;4. Prezo提供多种模板,自动生成图文并茂幻灯片,适合学生与初创团队。 如果您有一份内容完…

    2025年12月6日 软件教程
    000
  • Pages怎么协作编辑同一文档 Pages多人实时协作的流程

    首先启用Pages共享功能,点击右上角共享按钮并选择“添加协作者”,设置为可编辑并生成链接;接着复制链接通过邮件或社交软件发送给成员,确保其使用Apple ID登录iCloud后即可加入编辑;也可直接在共享菜单中输入邮箱地址定向邀请,设定编辑权限后发送;最后在共享面板中管理协作者权限,查看实时在线状…

    2025年12月6日 软件教程
    100
  • 咸鱼遇到“只退款不退货”的买家怎么办_咸鱼处理只退款不退货方法

    先与买家协商解决,要求其按规则退货退款,并保留聊天记录;若协商无效,申请平台介入并提交发货、签收及沟通等证据;若平台处理不利且金额较大,可依法提起民事诉讼,主张买家违反《民法典》合同规定,追回货款。 如果您在咸鱼平台出售手机后,买家申请“仅退款不退货”,这可能导致您既损失商品又损失资金。以下是应对该…

    2025年12月6日 软件教程
    000
  • 怎么下载安装快手极速版_快手极速版下载安装详细教程

    1、优先通过华为应用市场搜索“快手极速版”,确认开发者为北京快手科技有限公司后安装;2、若应用商店无结果,可访问快手极速版官网下载APK文件,需手动开启浏览器的未知来源安装权限;3、也可选择豌豆荚、应用宝等可信第三方平台下载官方版本,核对安全标识后完成安装。 如果您尝试在手机上安装快手极速版,但无法…

    2025年12月6日 软件教程
    000
  • 哔哩哔哩的视频卡在加载中怎么办_哔哩哔哩视频加载卡顿解决方法

    视频加载停滞可先切换网络或重启路由器,再清除B站缓存并重装应用,接着调低播放清晰度并关闭自动选分辨率,随后更改播放策略为AVC编码,最后关闭硬件加速功能以恢复播放。 如果您尝试播放哔哩哔哩的视频,但进度条停滞在加载状态,无法继续播放,这通常是由于网络、应用缓存或播放设置等因素导致。以下是解决此问题的…

    2025年12月6日 软件教程
    000
  • REDMI K90系列正式发布,售价2599元起!

    10月23日,redmi k90系列正式亮相,推出redmi k90与redmi k90 pro max两款新机。其中,redmi k90搭载骁龙8至尊版处理器、7100mah大电池及100w有线快充等多项旗舰配置,起售价为2599元,官方称其为k系列迄今为止最完整的标准版本。 图源:REDMI红米…

    2025年12月6日 行业动态
    200
  • Linux中如何安装Nginx服务_Linux安装Nginx服务的完整指南

    首先更新系统软件包,然后通过对应包管理器安装Nginx,启动并启用服务,开放防火墙端口,最后验证欢迎页显示以确认安装成功。 在Linux系统中安装Nginx服务是搭建Web服务器的第一步。Nginx以高性能、低资源消耗和良好的并发处理能力著称,广泛用于静态内容服务、反向代理和负载均衡。以下是在主流L…

    2025年12月6日 运维
    000
  • Linux journalctl与systemctl status结合分析

    先看 systemctl status 确认服务状态,再用 journalctl 查看详细日志。例如 nginx 启动失败时,systemctl status 显示 Active: failed,journalctl -u nginx 发现端口 80 被占用,结合两者可快速定位问题根源。 在 Lin…

    2025年12月6日 运维
    100
  • 华为新机发布计划曝光:Pura 90系列或明年4月登场

    近日,有数码博主透露了华为2025年至2026年的新品规划,其中pura 90系列预计在2026年4月发布,有望成为华为新一代影像旗舰。根据路线图,华为将在2025年底至2026年陆续推出mate 80系列、折叠屏新机mate x7系列以及nova 15系列,而pura 90系列则将成为2026年上…

    2025年12月6日 行业动态
    100
  • 菜鸟app的语音助手怎么唤醒_菜鸟app语音助手使用方法

    检查菜鸟App麦克风及后台运行权限;2. 在App内开启语音助手功能;3. 通过首页麦克风图标手动唤醒;4. 更新App至最新版本以确保功能正常。 如果您在使用菜鸟App时希望快速获取快递信息或执行相关操作,但发现语音助手无法响应,可能是由于唤醒功能未正确设置。以下是解决此问题的步骤: 本文运行环境…

    2025年12月6日 软件教程
    000
  • Linux如何优化系统性能_Linux系统性能优化的实用方法

    优化Linux性能需先监控资源使用,通过top、vmstat等命令分析负载,再调整内核参数如TCP优化与内存交换,结合关闭无用服务、选用合适文件系统与I/O调度器,持续按需调优以提升系统效率。 Linux系统性能优化的核心在于合理配置资源、监控系统状态并及时调整瓶颈环节。通过一系列实用手段,可以显著…

    2025年12月6日 运维
    000
  • 方正证券新股中签后怎么缴款_方正证券新股中签缴款教程

    中签后需在T+2日16:00前备足资金,方正证券将自动扣款。通过小方APP、短信或中签查询功能确认结果,缴款金额为中签股数×发行价,可用账户余额、卖股资金或银证转账充值,建议多存几十元作缓冲。系统通常于T+2日收盘后扣款,若资金不足或被其他自动交易占用导致失败,一年累计弃购3次将被限制半年打新。核心…

    2025年12月6日 软件教程
    000
  • E票电影app购票流程

    E票电影app使用指南: 1、安装完成后启动e票电影应用程序; 2、在首页的搜索框中输入你想观看的影片名称; Type Studio 一个视频编辑器,提供自动转录、自动生成字幕、视频翻译等功能 61 查看详情 3、选择场次后,点击“购票”按钮完成选座下单。 以上就是E票电影app购票流程的详细内容,…

    2025年12月6日 软件教程
    000
  • 爱聊app年龄修改入口

    爱聊app年龄修改入口: 1、打开app后,先点击界面右下角的“我”,然后点击顶部的个人“头像”; 2、进入个人资料页面后,点击右上角的“编辑”按钮; 3、在资料列表中找到“生日”选项,点击右侧显示的具体出生日期; 4、调整生日至正确的时间,修改完成后点击右上角的“确定”按钮,即可成功更新年龄信息。…

    2025年12月6日 软件教程
    000
  • 曝小米17 Air正在筹备 超薄机身+2亿像素+eSIM技术?

    近日,手机行业再度掀起超薄机型热潮,三星与苹果已相继推出s25 edge与iphone air等轻薄旗舰,引发市场高度关注。在此趋势下,多家国产厂商被曝正积极布局相关技术,加速抢占这一细分赛道。据业内人士消息,小米的超薄旗舰机型小米17 air已进入筹备阶段。 小米17 Pro 爆料显示,小米正在评…

    2025年12月6日 行业动态
    000
  • 「世纪传奇刀片新篇」飞利浦影音双11声宴开启

    百年声学基因碰撞前沿科技,一场有关声音美学与设计美学的影音狂欢已悄然引爆2025“双十一”! 当绝大多数影音数码品牌还在价格战中挣扎时,飞利浦影音已然开启了一场跨越百年的“声”活革命。作为拥有深厚技术底蕴的音频巨头,飞利浦影音及配件此次“双十一”精准聚焦“传承经典”与“设计美学”两大核心,为热爱生活…

    2025年12月6日 行业动态
    000
  • 荣耀手表5Pro 10月23日正式开启首销国补优惠价1359.2元起售

    荣耀手表5pro自9月25日开启全渠道预售以来,市场热度持续攀升,上市初期便迎来抢购热潮,一度出现全线售罄、供不应求的局面。10月23日,荣耀手表5pro正式迎来首销,提供蓝牙版与esim版两种选择。其中,蓝牙版本的攀登者(橙色)、开拓者(黑色)和远航者(灰色)首销期间享受国补优惠价,到手价为135…

    2025年12月6日 行业动态
    000

发表回复

登录后才能评论
关注微信