Java 8 のネストされた (マルチレベル) Group By
複数のキーに基づいてオブジェクトのマルチレベルのグループ化を実行する必要が生じることがよくあります。 。この場合、キーを持つ項目のリストと、同様にキーを持つサブ項目のリストを含むデータを表すクラスのセットがあります。目標は、項目をキーでグループ化し、集計ごとにサブ項目をキーでさらにグループ化し、ネストされたマップを作成することです。
Java 8 の Collectors.groupingBy を使用すると、以下の組み合わせを使用することでこれを実現できます。 flatMap とカスタム フラット マッピング コレクターの。まず、Stream. flatMap を使用して項目とサブ項目の組み合わせを保持する一時的なペアを作成します。
<code class="java">Map<T, Map<V, List<SubItem>>> result = pojo.getItems().stream() .flatMap(item -> item.subItems.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 9 まで利用可能です。このアプローチにより、コードは次のように簡素化されます。
<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 9 を待ちたくないユーザーのために、カスタム フラット マッピング コレクターを実装できます。
<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>
以上がJava 8 で Collectors.groupingBy を使用して Java でネストされたグループ化を実装する方法の詳細内容です。詳細については、PHP 中国語 Web サイトの他の関連記事を参照してください。