
本文详解如何使用 java 8+ stream api 对对象列表按指定属性(如 `id`)分组,并将每组内对象的特定字段(如 `value`)拼接为带标识的字符串,最终生成清晰、可读的结构化结果。
在实际开发中,常需将扁平化的对象列表按某一维度(如 id、category 或 type)聚合成逻辑分组,并进一步提取、转换每组数据——例如,将同一 id 下多个 Sample 对象的 value 字段按顺序拼接,末尾附加该 id,形成统一格式的字符串行。这种需求广泛见于日志聚合、报表生成、配置组装等场景。
以下以 Sample 记录类(record,推荐用于不可变数据载体)为例,展示完整、健壮的实现方案:
record Sample(String name, String value, String id) {}
// 示例输入数据
Sample[] samples = {
new Sample("title", "Title", "1"),
new Sample("input", "DCE", "1"),
new Sample("output", "CDE", "1"),
new Sample("title", "Title", "2"),
new Sample("input", "DCEE", "2"),
new Sample("output", "CDEE", "2"),
new Sample("title", "Title", "3"),
new Sample("input", "DCEEE", "3"),
new Sample("output", "CDEEE", "3")
};核心思路分为两步:
-
分组:使用 Collectors.groupingBy(Sample::id) 将 Sample 流按 id 聚合成 Map
>; -
转换与拼接:对每个分组条目(Map.Entry
>),提取所有 value 并用空格连接,再追加 id,最终收集为 List 。
推荐采用链式流式写法,简洁且语义清晰:
立即学习“Java免费学习笔记(深入)”;
import java.util.*; import java.util.stream.Collectors; ListgroupedOutput = Arrays.stream(samples) .collect(Collectors.groupingBy( Sample::id, // 按 id 分组 LinkedHashMap::new, // 保持插入顺序(可选,但利于可预测输出) Collectors.mapping(Sample::value, // 提取每项的 value Collectors.joining(" ")) // 拼接为 "Title DCE CDE" )) .entrySet().stream() .map(entry -> entry.getValue() + " " + entry.getKey()) // 拼入 id → "Title DCE CDE 1" .collect(Collectors.toList()); // 输出结果: // ["Title DCE CDE 1", "Title DCEE CDEE 2", "Title DCEEE CDEEE 3"]
✅ 关键优势说明:
- 使用 LinkedHashMap::new 作为 groupingBy 的 mapFactory 参数,确保分组结果按 id 首次出现顺序排列(如 1→2→3),避免因哈希随机性导致输出错乱;
- Collectors.mapping(..., Collectors.joining(" ")) 在分组阶段即完成 value 拼接,比先分组再遍历更高效;
- 整个流程无副作用、不可变,符合函数式编程原则,线程安全(前提是输入数据不可变)。
⚠️ 注意事项:
- 若原始 Sample 类为传统 class,请确保 id() 方法存在(或改用 sample.getId());若未重写 equals/hashCode,不影响本例(因仅依赖 String id 的自然比较);
- 若 value 可能为 null,建议在 mapping 前增加非空校验,例如 .map(s -> Optional.ofNullable(s.value()).orElse(""));
- 如需严格按 name 顺序(如 "title" → "input" → "output")拼接 value,应在分组后对 List
显式排序: .collect(Collectors.collectingAndThen( Collectors.toList(), list -> list.stream() .sorted(Comparator.comparing(Sample::name)) .map(Sample::value) .collect(Collectors.joining(" ")) ))
综上,该方案以声明式风格精准达成目标:将原始列表按 id 分组、提取并有序拼接 value、附加 id 标识,最终返回结构清晰、易于后续处理的字符串列表。掌握此模式,可快速适配各类基于字段聚合与格式化的需求。










