0

0

Java中递归过滤并插入复杂嵌套JSON到MongoDB的完整实践指南

霞舞

霞舞

发布时间:2026-02-26 15:04:15

|

943人浏览过

|

来源于php中文网

原创

Java中递归过滤并插入复杂嵌套JSON到MongoDB的完整实践指南

本文详解如何使用Java驱动递归解析深层嵌套JSON,自动跳过component值为containers/EmptyContainer.vue的对象,并将其结构化转换为符合业务需求的MongoDB文档(含_id、field、value三层结构),附可运行代码与关键注意事项。

本文详解如何使用java驱动递归解析深层嵌套json,自动跳过`component`值为`containers/emptycontainer.vue`的对象,并将其结构化转换为符合业务需求的mongodb文档(含`_id`、`field`、`value`三层结构),附可运行代码与关键注意事项。

在实际企业级应用中,前端路由配置、权限菜单树或动态工作区定义常以深度嵌套的JSON格式存储于文件或配置中心。当需将此类结构持久化至MongoDB时,不仅要求精准还原嵌套关系,还需支持条件过滤(如剔除占位组件)、结构映射(如将顶层键转为field字段)和类型安全转换(如布尔值、空字符串处理)。以下提供一套健壮、可扩展的Java实现方案。

Warp
Warp

新一代的终端工具(内置AI命令搜索)

下载

核心设计思路

  • 递归遍历 + 条件剪枝:对每个JSON对象检查component字段,若匹配"containers/EmptyContainer.vue"则整节点跳过,不递归其子节点;
  • 结构重塑:将原始JSON中顶层键(如"databases")作为field字段,其值作为value子文档,同时注入自动生成的_id;
  • 类型兼容性保障:使用org.json库解析,配合Document.append()自动处理String/Boolean/JSONArray/JSONObject等类型,避免手动强转异常;
  • 扁平化聚合输出:最终返回List,每项对应一个顶层模块(如databases),便于批量写入MongoDB。

完整可运行代码示例

import com.mongodb.client.MongoCollection;
import com.mongodb.client.MongoDatabase;
import com.mongodb.client.model.InsertManyOptions;
import org.bson.Document;
import org.json.JSONArray;
import org.json.JSONObject;

import java.io.IOException;
import java.nio.file.Files;
import java.nio.file.Paths;
import java.util.*;

public class NestedJsonToMongo {

    // 主入口:读取JSON文件 → 构建Document列表 → 批量插入
    public static void insertNestedJsonToMongo(String jsonFilePath, MongoCollection<Document> collection) 
            throws IOException {
        String jsonContent = Files.readString(Paths.get(jsonFilePath));
        JSONObject rootJson = new JSONObject(jsonContent);

        List<Document> documents = new ArrayList<>();
        // 遍历根对象的每个顶级字段(如 "databases")
        for (String topLevelKey : rootJson.keySet()) {
            Object topLevelValue = rootJson.get(topLevelKey);
            if (topLevelValue instanceof JSONObject) {
                Document doc = buildMongoDocument(topLevelKey, (JSONObject) topLevelValue);
                if (doc != null) { // 过滤后可能为null
                    documents.add(doc);
                }
            }
        }

        // 批量插入(启用有序写入,便于错误定位)
        if (!documents.isEmpty()) {
            collection.insertMany(documents, new InsertManyOptions().ordered(true));
            System.out.printf("✅ 成功插入 %d 个顶层模块到 MongoDB\n", documents.size());
        }
    }

    // 构建单个MongoDB文档:{_id: ..., field: "databases", value: {...}}
    private static Document buildMongoDocument(String fieldKey, JSONObject valueObj) {
        // Step 1: 过滤根节点 — 若其component为EmptyContainer.vue,则整个分支丢弃
        if (valueObj.has("component") && 
            "containers/EmptyContainer.vue".equals(valueObj.optString("component"))) {
            return null;
        }

        Document doc = new Document();
        doc.append("_id", UUID.randomUUID().toString()); // 或使用ObjectId.get()
        doc.append("field", fieldKey);
        doc.append("value", convertToBsonCompatible(valueObj));
        return doc;
    }

    // 递归转换JSONObject为BSON兼容结构(自动处理children数组过滤)
    private static Object convertToBsonCompatible(Object obj) {
        if (obj instanceof JSONObject) {
            JSONObject jsonObj = (JSONObject) obj;
            Document doc = new Document();

            // 复制所有非"children"字段
            for (String key : jsonObj.keySet()) {
                if ("children".equals(key)) continue;
                Object val = jsonObj.get(key);
                // 安全转换:处理null、布尔字符串等
                if (val instanceof Boolean || val instanceof Number || val instanceof String) {
                    doc.append(key, val);
                } else if (val == null) {
                    doc.append(key, null);
                }
            }

            // 特殊处理children:递归过滤并重建数组
            if (jsonObj.has("children")) {
                JSONArray childrenArray = jsonObj.getJSONArray("children");
                List<Document> filteredChildren = new ArrayList<>();
                for (int i = 0; i < childrenArray.length(); i++) {
                    Object child = childrenArray.get(i);
                    if (child instanceof JSONObject) {
                        JSONObject childObj = (JSONObject) child;
                        // 关键过滤逻辑:跳过component为EmptyContainer.vue的节点
                        if (!"containers/EmptyContainer.vue".equals(childObj.optString("component"))) {
                            filteredChildren.add((Document) convertToBsonCompatible(childObj));
                        }
                        // 注意:此处不递归进入被过滤节点的children,实现“剪枝”
                    }
                }
                if (!filteredChildren.isEmpty()) {
                    doc.append("children", filteredChildren);
                }
            }
            return doc;
        } else if (obj instanceof JSONArray) {
            JSONArray jsonArray = (JSONArray) obj;
            List<Object> list = new ArrayList<>();
            for (int i = 0; i < jsonArray.length(); i++) {
                list.add(convertToBsonCompatible(jsonArray.get(i)));
            }
            return list;
        } else {
            return obj; // 基础类型(String, Number, Boolean)直接返回
        }
    }

    // 示例调用(需替换为真实Mongo连接)
    public static void main(String[] args) {
        // MongoDatabase db = ... ; // 初始化你的MongoClient & Database
        // MongoCollection<Document> coll = db.getCollection("menu_configs");

        try {
            // insertNestedJsonToMongo("config.json", coll);
            System.out.println("? 示例:已生成待插入文档结构(模拟输出):");
            JSONObject sample = new JSONObject("""
                {"databases": {"component":"pages/.../DatasourcesWorkspace.vue","children":[
                  {"name":"Databases","path":"databases","component":"pages/.../Databases.vue","hide":true},
                  {"name":"Schemas","path":"schemas","component":"containers/EmptyContainer.vue","hide":true,"children":[...]}
                ]}}
                """);
            Document doc = buildMongoDocument("databases", sample.getJSONObject("databases"));
            System.out.println(doc.toJson());
        } catch (Exception e) {
            e.printStackTrace();
        }
    }
}

关键注意事项与最佳实践

  • ? 空值与缺失字段防护:始终使用optString()、optBoolean()替代getString()/getBoolean(),避免JSONException;对可能为null的字段(如path、help)显式判断后再append;
  • ? 嵌套深度无限制:本方案采用纯递归+JSON库原生类型识别,天然支持任意深度嵌套,无需预设层级;
  • ⚙️ 性能优化建议:对超大JSON文件(>10MB),可改用流式解析器(如Jackson Streaming API)替代org.json,减少内存占用;
  • ? Schema灵活性:答案中提及的“schema可优化”值得重视——若业务允许,建议将value展开为扁平化字段(如value_component, value_children_count),提升查询效率;
  • ? ID生成策略:示例使用UUID,生产环境推荐ObjectId.get()以获得时间序+唯一性+紧凑性三重优势;
  • ? 单元测试必备:务必覆盖边界用例:空children数组、缺失component字段、name为空字符串、深层嵌套过滤链等。

通过以上方案,开发者可稳定、清晰地将复杂前端配置JSON转化为MongoDB中的结构化数据,兼顾可维护性与运行时鲁棒性。

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
json数据格式
json数据格式

JSON是一种轻量级的数据交换格式。本专题为大家带来json数据格式相关文章,帮助大家解决问题。

449

2023.08.07

json是什么
json是什么

JSON是一种轻量级的数据交换格式,具有简洁、易读、跨平台和语言的特点,JSON数据是通过键值对的方式进行组织,其中键是字符串,值可以是字符串、数值、布尔值、数组、对象或者null,在Web开发、数据交换和配置文件等方面得到广泛应用。本专题为大家提供json相关的文章、下载、课程内容,供大家免费下载体验。

546

2023.08.23

jquery怎么操作json
jquery怎么操作json

操作的方法有:1、“$.parseJSON(jsonString)”2、“$.getJSON(url, data, success)”;3、“$.each(obj, callback)”;4、“$.ajax()”。更多jquery怎么操作json的详细内容,可以访问本专题下面的文章。

325

2023.10.13

go语言处理json数据方法
go语言处理json数据方法

本专题整合了go语言中处理json数据方法,阅读专题下面的文章了解更多详细内容。

81

2025.09.10

string转int
string转int

在编程中,我们经常会遇到需要将字符串(str)转换为整数(int)的情况。这可能是因为我们需要对字符串进行数值计算,或者需要将用户输入的字符串转换为整数进行处理。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

850

2023.08.02

java中boolean的用法
java中boolean的用法

在Java中,boolean是一种基本数据类型,它只有两个可能的值:true和false。boolean类型经常用于条件测试,比如进行比较或者检查某个条件是否满足。想了解更多java中boolean的相关内容,可以阅读本专题下面的文章。

364

2023.11.13

java boolean类型
java boolean类型

本专题整合了java中boolean类型相关教程,阅读专题下面的文章了解更多详细内容。

40

2025.11.30

c语言中null和NULL的区别
c语言中null和NULL的区别

c语言中null和NULL的区别是:null是C语言中的一个宏定义,通常用来表示一个空指针,可以用于初始化指针变量,或者在条件语句中判断指针是否为空;NULL是C语言中的一个预定义常量,通常用来表示一个空值,用于表示一个空的指针、空的指针数组或者空的结构体指针。

248

2023.09.22

Golang 实际项目案例:从需求到上线
Golang 实际项目案例:从需求到上线

《Golang 实际项目案例:从需求到上线》以真实业务场景为主线,完整覆盖需求分析、架构设计、模块拆分、编码实现、性能优化与部署上线全过程,强调工程规范与实践决策,帮助开发者打通从技术实现到系统交付的关键路径,提升独立完成 Go 项目的综合能力。

1

2026.02.26

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
Kotlin 教程
Kotlin 教程

共23课时 | 3.9万人学习

C# 教程
C# 教程

共94课时 | 10.2万人学习

Java 教程
Java 教程

共578课时 | 72.8万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号