Java 8 中使用 Nested GroupBy 进行多级分组
本文探讨了如何在处理嵌套类时实现多级分组Java 8。具体来说,目标是按 key1 字段对项目进行分组,然后对于每组项目,进一步按 key2 字段对它们进行分组。最终,输出应该是一个以 key1 作为外键的映射,以及一个 key2 到子项列表的映射。
最初的方法尝试使用 Collectors.groupingBy 方法来实现这一点,但是,它是无法直接通过多个键对单个项目进行分组。为了克服这个问题,使用了 flatMap 操作。
一种方法是使用 Map.entrySet 创建一个临时对,以在收集之前保存项目和子项目的组合。 Java 9 中提供的另一种方法利用了 flatMapping 收集器,它可以直接在收集器中执行 flatMap 操作。
这是 flatMap 解决方案:
<code class="java">Map<T, Map<V, List<SubItem>>> result = pojo.getItems().stream() .flatMap(item -> item.getSubItems().stream() .map(sub -> new AbstractMap.SimpleImmutableEntry<>(item.getKey1(), sub))) .collect(Collectors.groupingBy(AbstractMap.SimpleImmutableEntry::getKey, Collectors.mapping(Map.Entry::getValue, Collectors.groupingBy(SubItem::getKey2))));</code>
使用自定义的替代方案Java 8 中的收集器:
<code class="java">static <T,U,A,R> Collector<T,?,R> flatMapping( Function<? super T,? extends Stream<? extends U>> mapper, Collector<? super U,A,R> downstream) { BiConsumer<A, ? super U> acc = downstream.accumulator(); return Collector.of(downstream.supplier(), (a, t) -> { try(Stream<? extends U> s=mapper.apply(t)) { if(s!=null) s.forEachOrdered(u -> acc.accept(a, u)); }}, downstream.combiner(), downstream.finisher(), downstream.characteristics().toArray(new Collector.Characteristics[0])); }</code>
此自定义收集器可以按如下方式使用:
<code class="java">Map<T, Map<V, List<SubItem>>> result = pojo.getItems().stream() .collect(Collectors.groupingBy(Item::getKey1, Collectors.flatMapping(item -> item.getSubItems().stream(), Collectors.groupingBy(SubItem::getKey2))));</code>
以上是如何在Java 8中实现嵌套对象的多级分组?的详细内容。更多信息请关注PHP中文网其他相关文章!