Java Stream API:数据分组与嵌套JSON列表转换实践

Java Stream API:数据分组与嵌套JSON列表转换实践

本文详细讲解如何利用Java Stream API将扁平化的数据列表(如数据库查询结果)进行高效分组,并将其转换为包含嵌套列表的复杂对象结构,最终便于序列化为JSON格式。通过groupingBy和mapping等操作,实现数据聚合与重塑,提升代码的简洁性和可读性。

引言

在现代应用开发中,我们经常需要处理来自数据库或其他数据源的扁平化数据列表。然而,为了满足前端展示、api接口或下游系统的数据消费需求,这些数据往往需要被重塑为更具结构化、层次化的形式,例如按某个字段分组并将其相关属性聚合为一个嵌套列表。java 8引入的stream api为这种复杂的数据转换提供了强大而优雅的解决方案,极大地简化了代码并提高了可读性。本教程将深入探讨如何利用stream api实现这一目标,最终输出符合特定嵌套结构的json数据。

数据模型定义

首先,我们根据需求定义原始数据模型和目标数据模型。

1. 原始响应类 (Response)

这通常是数据库查询结果映射到的对象,包含一个主键ID和两个描述性字段。

public interface Response {    Long getId();    String getSData();    String getSName();}// 为了方便演示和创建实例,我们提供一个实现类public static class ResponseImpl implements Response {    private Long id;    private String sData;    private String sName;    public ResponseImpl(Long id, String sData, String sName) {        this.id = id;        this.sData = sData;        this.sName = sName;    }    @Override    public Long getId() { return id; }    @Override    public String getSData() { return sData; }    @Override    public String getSName() { return sName; }    @Override    public String toString() {        return "ResponseImpl{id=" + id + ", sData='" + sData + "', sName='" + sName + "'}";    }}

2. 子数据类 (SubData)

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

这是分组后嵌套在主对象中的列表元素,它包含了原始Response中除id以外的两个字段。

public static class SubData {    private String sData;    private String sName;    public SubData(String sData, String sName) {        this.sData = sData;        this.sName = sName;    }    // Getters and Setters for JSON serialization    public String getSData() { return sData; }    public void setSData(String sData) { this.sData = sData; }    public String getSName() { return sName; }    public void setSName(String sName) { this.sName = sName; }    @Override    public String toString() {        return "SubData{sData='" + sData + "', sName='" + sName + "'}";    }}

3. 目标响应类 (NewResponse)

这是我们希望最终得到的数据结构,它包含原始的id和一个SubData对象的列表。

public static class NewResponse {    private Long id;    private List subDataList;    public NewResponse(Long id, List subDataList) {        this.id = id;        this.subDataList = subDataList;    }    // Getters and Setters for JSON serialization    public Long getId() { return id; }    public void setId(Long id) { this.id = id; }    public List getSubDataList() { return subDataList; }    public void setSubDataList(List subDataList) { this.subDataList = subDataList; }    @Override    public String toString() {        return "NewResponse{id=" + id + ", subDataList=" + subDataList + "}";    }}

原始数据准备

为了演示,我们创建一份模拟的原始Response列表,其结构与问题描述中的JSON数据一致。

import java.util.Arrays;import java.util.List;import java.util.Map;import java.util.stream.Collectors;public class DataGroupingTutorial {    // ... (定义上面提到的 ResponseImpl, SubData, NewResponse 类) ...    public static void main(String[] args) {        List responses = Arrays.asList(            new ResponseImpl(1L, "UK", "X"),            new ResponseImpl(1L, "FR", "X"), // 修正:原始数据中sData和sName是分开的,这里模拟成这样            new ResponseImpl(2L, "UK", "Y"),            new ResponseImpl(2L, "FR", "Y"),            new ResponseImpl(4L, "EU", "X"),            new ResponseImpl(4L, "Others", "O")        );        System.out.println("原始数据:");        responses.forEach(System.out::println);        System.out.println("n---");    }}

注:根据原始问题描述的JSON示例 {“id”:1,”sData”:”UK,FR”,”sName”:”X},sData字段似乎包含了多个值。但在Response接口中sData是String类型,且期望输出的SubData中sData也是单个值。为了符合教程的通用性,这里假设sData和sName是单个值,若sData本身就是逗号分隔的字符串,则在SubData构造时需要额外处理。本教程以最直接的映射方式进行。

核心转换逻辑

我们将分两步使用Stream API来完成数据转换。

1. 步骤一:初步分组与子数据映射

这一步的目标是将原始List按照id进行分组,并且在分组的同时,将每个Response对象转换为SubData对象,并收集成一个列表。

// ... (接上面的main方法) ...        Map<Long, List> grouped = responses.stream()                .collect(Collectors.groupingBy(                        Response::getId, // 按Response的ID进行分组                        Collectors.mapping(                                r -> new SubData(r.getSData(), r.getSName()), // 将每个Response映射为SubData                                Collectors.toList() // 将映射后的SubData收集成一个列表                        )                ));        System.out.println("步骤一:分组后的中间结果 (Map<Long, List>):");        grouped.forEach((id, subDataList) -> System.out.println("ID: " + id + ", SubDataList: " + subDataList));        System.out.println("n---");

Collectors.groupingBy(Response::getId, …): 这是Stream API中用于分组的核心收集器。它接收一个分类函数(这里是Response::getId,表示按id字段分组),以及一个下游收集器。Collectors.mapping(r -> new SubData(r.getSData(), r.getSName()), Collectors.toList()): 这是groupingBy的下游收集器。Collectors.mapping(…):在分组内部,将每个Response对象通过提供的映射函数r -> new SubData(r.getSData(), r.getSName())转换为一个新的SubData对象。Collectors.toList():将所有转换后的SubData对象收集到一个新的List中。

经过这一步,我们得到了一个Map<Long, List>,其中key是原始Response的id,value是该id下所有对应的SubData对象的列表。

2. 步骤二:将分组结果转换为目标对象列表

现在我们有了一个Map<Long, List>,我们需要将其转换为List。这可以通过遍历Map的entrySet()并进行映射来完成。

// ... (接上面的main方法) ...        List finalResult = grouped.entrySet()                .stream() // 将Map的EntrySet转换为Stream                .map(entry -> new NewResponse(entry.getKey(), entry.getValue())) // 将每个Entry映射为NewResponse对象                .collect(Collectors.toList()); // 收集成List        System.out.println("步骤二:最终结果 (List):");        finalResult.forEach(System.out::println);        System.out.println("n---");

grouped.entrySet().stream(): 获取Map的键值对集合(Set<Map.Entry<Long, List>>),并将其转换为Stream。map(entry -> new NewResponse(entry.getKey(), entry.getValue())): 对Stream中的每个Map.Entry对象进行映射。entry.getKey()提供了id,entry.getValue()提供了List,正好可以用来构造NewResponse对象。collect(Collectors.toList()): 将所有映射后的NewResponse对象收集到一个新的List中。

至此,我们成功地将扁平化的List转换成了List,其结构完全符合我们预期的嵌套JSON格式。

链式操作优化 (可选)

如果你希望代码更简洁,可以将上述两个步骤合并为一个Stream链。虽然这会减少中间变量,但对于非常复杂的逻辑,可能会略微降低可读性。

// ... (接上面的main方法) ...        System.out.println("链式操作优化后的结果 (List):");        List chainedResult = responses.stream()                .collect(Collectors.groupingBy(                        Response::getId,                        Collectors.mapping(                                r -> new SubData(r.getSData(), r.getSName()),                                Collectors.toList()                        )                )) // 结果是 Map<Long, List>                .entrySet() // 获取Map的EntrySet                .stream() // 将EntrySet转换为Stream                .map(entry -> new NewResponse(entry.getKey(), entry.getValue())) // 映射为NewResponse                .collect(Collectors.toList()); // 收集最终结果        chainedResult.forEach(System.out::println);        System.out.println("n---");

完整示例代码

import java.util.Arrays;import java.util.List;import java.util.Map;import java.util.stream.Collectors;public class DataGroupingTutorial {    // 原始响应接口    public interface Response {        Long getId();        String getSData();        String getSName();    }    // 原始响应接口的实现类    public static class ResponseImpl implements Response {        private Long id;        private String sData;        private String sName;        public ResponseImpl(Long id, String sData, String sName) {            this.id = id;            this.sData = sData;            this.sName = sName;        }        @Override        public Long getId() { return id; }        @Override        public String getSData() { return sData; }        @Override        public String getSName() { return sName; }        @Override        public String toString() {            return "ResponseImpl{id=" + id + ", sData='" + sData + "', sName='" + sName + "'}";        }    }    // 子数据类    public static class SubData {        private String sData;        private String sName;        public SubData(String sData, String sName) {            this.sData = sData;            this.sName = sName;        }        // Getters and Setters for JSON serialization        public String getSData() { return sData; }        public void setSData(String sData) { this.sData = sData; }        public String getSName() { return sName; }        public void setSName(String sName) { this.sName = sName; }        @Override        public String toString() {            return "SubData{sData='" + sData + "', sName='" + sName + "'}";        }    }    // 目标响应类    public static class NewResponse {        private Long id;        private List subDataList;        public NewResponse(Long id, List subDataList) {            this.id = id;            this.subDataList = subDataList;        }        // Getters and Setters for JSON serialization        public Long getId() { return id; }        public void setId(Long id) { this.id = id; }        public List getSubDataList() { return subDataList; }        public void setSubDataList(List subDataList) { this.subDataList = subDataList; }        @Override        public String toString() {            return "NewResponse{id=" + id + ", subDataList=" + subDataList + "}";        }    }    public static void main(String[] args) {        // 模拟原始数据        List responses = Arrays.asList(            new ResponseImpl(1L, "UK", "X"),            new ResponseImpl(1L, "FR", "X"),            new ResponseImpl(2L, "UK", "Y"),            new ResponseImpl(2L, "FR", "Y"),            new ResponseImpl(4L, "EU", "X"),            new ResponseImpl(4L, "Others", "O")        );        System.out.println("原始数据:");        responses.forEach(System.out::println);        System.out.println("n---");        // 步骤一:初步分组与子数据映射        Map<Long, List> grouped = responses.

以上就是Java Stream API:数据分组与嵌套JSON列表转换实践的详细内容,更多请关注创想鸟其它相关文章!

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

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2025年11月15日 12:06:03
下一篇 2025年11月15日 12:27:25

相关推荐

  • Golang HTTP请求路由性能优化示例

    使用高效路由库如httprouter可显著提升Go服务性能,其基于Radix Tree实现快速精确匹配,支持动态参数与通配符,避免反射和动态分配,性能优于标准mux;结合中间件精简、路径匹配顺序优化及pprof分析,可有效降低延迟,提升高并发场景下的请求处理效率。 在Go语言中构建高性能的HTTP服…

    好文分享 2025年12月16日
    000
  • Golang TCP服务器并发请求处理示例

    Go语言通过goroutine和net包实现高并发TCP服务器,每个连接由独立goroutine处理,互不阻塞;使用net.Listen监听端口,Accept接收连接,每新连接启goroutine通信;示例代码展示服务端读取客户端消息并回复;Go轻量级goroutine结合I/O多路复用实现高效并发…

    2025年12月16日
    000
  • 优雅地中断 io.CopyN 操作:一种实用指南

    本文旨在介绍如何优雅地中断 io.CopyN 函数的执行。io.CopyN 常用于从一个 io.Reader 复制指定数量的字节到 io.Writer。本文将通过关闭输入文件的方法,演示如何提前终止 io.CopyN 的操作,并提供相应的代码示例和注意事项,帮助开发者在实际应用中灵活应对类似场景。 …

    2025年12月16日
    000
  • Golang多模块项目引用关系管理实践

    采用单仓库多模块结构,通过Go Module的replace指令管理本地依赖,结合清晰的目录划分与接口解耦,避免循环依赖,利用自动化工具统一维护依赖和版本,确保各模块可独立构建测试,降低项目复杂度。 在Golang多模块项目中,模块间的引用关系如果管理不当,很容易导致版本混乱、依赖冲突或构建失败。核…

    2025年12月16日
    000
  • Golang中通过Stdin向命令传递数据并从Stdout接收数据

    本文介绍了如何在Golang中使用os/exec包执行外部命令,并通过Stdin向命令传递数据,同时从Stdout读取命令的输出。文章通过示例代码展示了如何正确地处理并发,避免常见的管道阻塞问题,确保数据能够完整地传递和接收。此外,还探讨了使用sync.WaitGroup来同步goroutine,以…

    2025年12月16日
    000
  • Go HTML 模板中 ZgotmplZ 错误的解析与安全实践

    在 Go HTML 模板渲染过程中,ZgotmplZ 值的出现表明存在潜在的安全风险,通常是由于不安全的字符串内容被注入到 HTML 属性或内容上下文。本文将深入解析 ZgotmplZ 的含义,并提供使用 html/template 包中 template.HTMLAttr 和 template.H…

    2025年12月16日
    000
  • Go语言中结构体如何正确引用数组切片:深入理解数组与切片的关系

    本文旨在阐明Go语言中结构体如何正确包含对数组的切片引用。针对将数组指针直接赋值给切片字段的常见错误,文章深入解析了Go切片与数组指针的根本区别。通过提供清晰的示例代码,教程演示了如何利用切片表达式array[:]从数组创建切片,并将其安全地赋值给结构体内的切片类型字段,以实现预期的引用行为。 Go…

    2025年12月16日
    000
  • Go语言:理解结构体中数组与切片的正确用法

    本文旨在阐明Go语言中结构体如何正确地引用数组作为切片字段。Go切片并非简单的数组指针,其内部包含指针、长度和容量。直接将数组的指针赋值给切片字段会导致类型不匹配错误。正确的做法是使用 array[:] 语法,将数组转换为一个切片视图,从而实现结构体对底层数组的有效引用。 Go语言切片(Slice)…

    2025年12月16日
    000
  • 如何在 Go 语言的结构体中使用指向数组的指针

    本文旨在阐述如何在 Go 语言的结构体中正确使用数组的指针或切片。通过示例代码和详细解释,帮助读者理解切片和数组指针的区别,以及如何在结构体中正确地定义和使用它们,避免常见的类型转换错误。 在 Go 语言中,结构体可以包含指向数组的指针或切片。然而,需要注意的是,切片(slice)并非简单的数组指针…

    2025年12月16日
    000
  • Golang包循环依赖检测与优化技巧

    包循环依赖指两个或多个包相互导入形成闭环,导致编译失败。可通过go list、go-depvis等工具检测并利用提取公共子包、依赖倒置、接口抽象等方式打破循环,结合分层架构与单一职责原则预防问题。 Go语言虽然在设计上避免了很多传统语言的复杂性,但随着项目规模扩大,包之间的依赖关系容易变得错综复杂,…

    2025年12月16日
    000
  • Golang 文件IO操作与性能优化实践

    合理使用Go标准库并优化IO策略可显著提升文件处理性能。1. 使用bufio减少系统调用,适合小块读写;2. 大文件用流式读取避免OOM,小文件可一次性加载;3. 并发分片读取大文件并配合预读提升吞吐;4. 结合系统调优如O_DIRECT、关闭atime等防止IO瓶颈。 Go语言在文件IO操作上提供…

    2025年12月16日
    000
  • Golang类型断言语法与接口使用技巧

    接口与类型断言用于实现Go语言的多态与类型安全操作。接口定义方法集,任何实现这些方法的类型自动满足该接口;空接口interface{}可存储任意类型值,常用于不确定类型的场景。使用类型断言value, ok := interfaceVar.(ConcreteType)可安全提取具体类型,避免pani…

    2025年12月16日
    000
  • Golang中通过Stdin传递数据并从Stdout接收数据

    本文旨在解决在Golang中,如何正确地将数据通过标准输入(stdin)传递给一个命令,并从该命令的标准输出(stdout)接收数据的常见问题。通过使用os/exec包,结合io.Copy和sync.WaitGroup,可以避免常见的race condition问题,确保数据的完整性和程序的稳定性。…

    2025年12月16日
    000
  • 优雅地中断 io.CopyN 操作:Go 语言实践教程

    在 Go 语言中,io.CopyN 函数是一个高效的数据复制工具,常用于将数据从一个 io.Reader 复制到 io.Writer。 然而,在某些场景下,我们可能需要在复制过程中途停止操作。 例如,当从网络连接或文件读取数据时,如果客户端断开连接或文件变得不可用,我们可能需要立即停止复制。本文将探…

    2025年12月16日
    000
  • 使用 PTY 实现 Go 程序与子进程的双向通信

    本文介绍了如何使用 PTY (Pseudo Terminal) 在 Go 程序中与子进程进行双向通信。传统管道方式在处理带有终端输出清除或输入缓冲的程序时会遇到问题,而 PTY 模拟终端环境,可以有效解决这些问题,实现更可靠的进程间通信。文章将详细讲解 PTY 的原理,并提供使用 github.co…

    2025年12月16日
    000
  • Golang测试断言库自定义函数实践

    自定义断言函数可提升Go测试的可读性与维护性,通过封装复杂逻辑、减少重复代码,支持如结构体验证、浮点比较等场景,结合testify与泛型实现高效断言。 在Go语言的测试实践中,使用断言库能显著提升代码可读性和测试效率。虽然标准库testing已经足够基础使用,但为了更简洁地表达期望结果,开发者常引入…

    2025年12月16日
    000
  • Golang包导入路径自动补全与优化技巧

    启用编辑器Go插件并配置gopls实现自动补全与导入;2. 使用goimports工具格式化代码、删除未使用包并自动修复导入;3. 基于Go Modules组织导入路径,确保项目可移植;4. 通过别名简化复杂导入,提升可读性。 在Go语言开发中,包导入路径的手动管理容易出错且影响效率。借助工具和规范…

    2025年12月16日
    000
  • Go 模板中访问外部作用域

    在使用 Go 模板时,with 和 range 语句会改变当前的作用域,这有时会使访问外部作用域的变量变得困难。本文将介绍如何在使用 with 或 range 语句时访问外部作用域,从而更灵活地使用 Go 模板。 当在 with 或 range 语句内部时,. 符号代表当前作用域的上下文。例如,在 …

    2025年12月16日
    000
  • 如何在 Go 模板的 “with” 或 “range” 作用域内访问外部作用域?

    本文旨在解决在使用 Go 模板时,如何在 with 或 range 语句创建的内部作用域中访问外部作用域的问题。通过使用 $ 符号,可以轻松访问模板执行的根数据对象,从而访问外部作用域中的变量和字段。本文将通过示例代码详细说明 $ 的用法。 在使用 Go 的 text/template 或 html…

    2025年12月16日
    000
  • Golang Helm Chart模板创建与管理

    使用Go语言辅助Helm Chart模板的创建与管理,通过官方库加载、渲染和验证Chart,结合CI/CD实现自动化版本发布与安全校验,提升Kubernetes应用部署效率。 使用Golang开发Helm Chart模板的创建与管理,通常结合代码生成工具和CI/CD流程来提升效率。虽然Helm本身基…

    2025年12月16日
    000

发表回复

登录后才能评论
关注微信