0

0

Java 中使用 Stream 和递归实现嵌套字符串的层级分组(如点号分隔路径)

霞舞

霞舞

发布时间:2026-01-28 21:08:16

|

581人浏览过

|

来源于php中文网

原创

java 中使用 stream 和递归实现嵌套字符串的层级分组(如点号分隔路径)

本文介绍如何将形如 "caso.responsavel.dadosPessoais.nome" 的点号分隔字符串列表,递归构建成嵌套的 `Map` 层级结构,并支持任意深度的路径解析与可视化输出。

在 Java 开发中,常需将扁平化的路径式字符串(如数据库字段映射、JSON Schema 路径或 DTO 层级属性)转化为直观的树状结构。例如,将 ["caso.id", "caso.responsavel.dadosPessoais.nome"] 自动组织为:

caso
   id
   responsavel
      dadosPessoais
         nome

这无法通过单层 Collectors.groupingBy() 实现,而需结合递归建模 + 动态嵌套 Map。下面提供一套简洁、可复用、类型安全的解决方案。

BibiGPT-哔哔终结者
BibiGPT-哔哔终结者

B站视频总结器-一键总结 音视频内容

下载

✅ 核心思路

  • 将每个字符串按 "\\." 拆分为路径节点数组(如 "caso.responsavel.nome" → ["caso", "responsavel", "nome"]);
  • 使用递归方法 fill(Map, String[] nodes, int index) 逐层插入:
    • 若当前节点(nodes[i])尚不存在,则新建子 HashMap 并挂载;
    • 若已存在且为 Map,则递归进入该子 Map 继续处理后续节点;
    • 到达末尾(i == nodes.length)时终止,无需额外存储值(仅建结构);
  • 最终得到一个 Map,其 value 可能是 Map(中间节点)或 null(叶子节点占位,实际可扩展为存储元数据)。

✅ 完整可运行代码

import java.util.*;

public class NestedGrouping {

    public static void main(String[] args) {
        String[] array = {
            "caso.id",
            "caso.unidadeDoCaso.id",
            "caso.etiqueta",
            "caso.sigiloso",
            "caso.idPecaSegredoJustica",
            "caso.numeroAno",
            "caso.numero",
            "caso.competencia.id",
            "caso.competencia.ativo",
            "caso.competencia.nome",
            "caso.responsavel.id",
            "caso.responsavel.dadosPessoais.nome",
            "caso.escrivao.id",
            "caso.escrivao.dadosPessoais.nome"
        };

        Map root = new HashMap<>();
        for (String path : array) {
            String[] nodes = path.split("\\.");
            fill(root, nodes, 0);
        }

        print(root, "");
    }

    // 递归构建嵌套 Map 结构
    public static void fill(Map map, String[] nodes, int i) {
        if (i >= nodes.length) return;

        String key = nodes[i];
        Object existing = map.get(key);

        if (existing == null) {
            // 创建新子 Map 并挂载
            Map child = new HashMap<>();
            map.put(key, child);
            fill(child, nodes, i + 1);
        } else if (existing instanceof Map) {
            // 已存在子 Map,继续递归
            @SuppressWarnings("unchecked")
            Map child = (Map) existing;
            fill(child, nodes, i + 1);
        }
        // ⚠️ 注意:若 existing 非 null 且非 Map(如冲突路径),此处可抛异常或忽略(本例视为合法结构)
    }

    // 递归打印,带缩进层次
    public static void print(Map map, String indent) {
        for (String key : map.keySet()) {
            System.out.println(indent + key);
            Object value = map.get(key);
            if (value instanceof Map) {
                print((Map) value, indent + "   ");
            }
        }
    }
}

✅ 输出效果(节选)

caso
   id
   unidadeDoCaso
      id
   etiqueta
   sigiloso
   idPecaSegredoJustica
   numeroAno
   numero
   competencia
      id
      ativo
      nome
   responsavel
      id
      dadosPessoais
         nome
   escrivao
      id
      dadosPessoais
         nome

⚠️ 注意事项与优化建议

  • 类型安全性:Object 作为 value 类型虽灵活,但建议封装为泛型工具类(如 NestedPathTree)或使用 Map 显式建模;
  • 重复路径处理:当前逻辑对重复路径(如两次 "caso.id")静默忽略;如需校验,可在 fill() 中添加 if (i == nodes.length - 1 && existing != null) 报警;
  • 空节点防护:split("\\.") 可能产生空字符串(如 "a..b"),建议预处理:Arrays.stream(nodes).filter(s -> !s.isEmpty()).toArray(String[]::new);
  • Stream 替代写法(非必须):虽然问题提到 Stream,但递归构建本质是副作用操作;若坚持函数式风格,可用 Arrays.stream(array).forEach(...) 替代 for-loop,语义等价;
  • 生产环境增强:可扩展支持 JSON 序列化(如 Jackson 的 @JsonAnyGetter)、路径查询(get("caso.responsavel.dadosPessoais"))、或导出为 YAML/TreeModel。

此方案轻量、无外部依赖、逻辑清晰,适用于权限字段过滤、动态表单生成、API 响应结构推导等典型场景。

立即学习Java免费学习笔记(深入)”;

热门AI工具

更多
DeepSeek
DeepSeek

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

豆包大模型
豆包大模型

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

通义千问
通义千问

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

腾讯元宝
腾讯元宝

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

文心一言
文心一言

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

讯飞写作
讯飞写作

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

即梦AI
即梦AI

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

ChatGPT
ChatGPT

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

相关专题

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

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

418

2023.08.07

json是什么
json是什么

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

535

2023.08.23

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

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

311

2023.10.13

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

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

77

2025.09.10

string转int
string转int

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

443

2023.08.02

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

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

236

2023.09.22

java中null的用法
java中null的用法

在Java中,null表示一个引用类型的变量不指向任何对象。可以将null赋值给任何引用类型的变量,包括类、接口、数组、字符串等。想了解更多null的相关内容,可以阅读本专题下面的文章。

438

2024.03.01

if什么意思
if什么意思

if的意思是“如果”的条件。它是一个用于引导条件语句的关键词,用于根据特定条件的真假情况来执行不同的代码块。本专题提供if什么意思的相关文章,供大家免费阅读。

776

2023.08.22

俄罗斯Yandex引擎入口
俄罗斯Yandex引擎入口

2026年俄罗斯Yandex搜索引擎最新入口汇总,涵盖免登录、多语言支持、无广告视频播放及本地化服务等核心功能。阅读专题下面的文章了解更多详细内容。

158

2026.01.28

热门下载

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

精品课程

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

共23课时 | 3万人学习

C# 教程
C# 教程

共94课时 | 7.8万人学习

Java 教程
Java 教程

共578课时 | 52.5万人学习

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

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