0

0

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

DDD

DDD

发布时间:2025-09-13 12:03:36

|

467人浏览过

|

来源于php中文网

原创

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> 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>):");
        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>,其中key是原始Response的id,value是该id下所有对应的SubData对象的列表。

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

现在我们有了一个Map>,我们需要将其转换为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>>),并将其转换为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>
                .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> grouped = responses.

热门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中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

422

2023.08.02

js 字符串转数组
js 字符串转数组

js字符串转数组的方法:1、使用“split()”方法;2、使用“Array.from()”方法;3、使用for循环遍历;4、使用“Array.split()”方法。本专题为大家提供js字符串转数组的相关的文章、下载、课程内容,供大家免费下载体验。

298

2023.08.03

js截取字符串的方法
js截取字符串的方法

js截取字符串的方法有substring()方法、substr()方法、slice()方法、split()方法和slice()方法。本专题为大家提供字符串相关的文章、下载、课程内容,供大家免费下载体验。

212

2023.09.04

java基础知识汇总
java基础知识汇总

java基础知识有Java的历史和特点、Java的开发环境、Java的基本数据类型、变量和常量、运算符和表达式、控制语句、数组和字符串等等知识点。想要知道更多关于java基础知识的朋友,请阅读本专题下面的的有关文章,欢迎大家来php中文网学习。

1498

2023.10.24

Python 自然语言处理(NLP)基础与实战
Python 自然语言处理(NLP)基础与实战

本专题系统讲解 Python 在自然语言处理(NLP)领域的基础方法与实战应用,涵盖文本预处理(分词、去停用词)、词性标注、命名实体识别、关键词提取、情感分析,以及常用 NLP 库(NLTK、spaCy)的核心用法。通过真实文本案例,帮助学习者掌握 使用 Python 进行文本分析与语言数据处理的完整流程,适用于内容分析、舆情监测与智能文本应用场景。

9

2026.01.27

热门下载

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

精品课程

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

共23课时 | 2.9万人学习

C# 教程
C# 教程

共94课时 | 7.7万人学习

Java 教程
Java 教程

共578课时 | 51.8万人学习

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

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