如何使用java8按谓词对列表进行分区?

2022-09-01 11:50:24

我有一个列表,我想拆分成几个小列表。a

说出所有包含“aaa”的项目,所有包含“bbb”的项目以及更多谓词。

我如何使用java8做到这一点?

我看到这篇文章但它只分成2个列表。

public void partition_list_java8() {

    Predicate<String> startWithS = p -> p.toLowerCase().startsWith("s");

    Map<Boolean, List<String>> decisionsByS = playerDecisions.stream()
            .collect(Collectors.partitioningBy(startWithS));

    logger.info(decisionsByS);

    assertTrue(decisionsByS.get(Boolean.TRUE).size() == 3);
}

我看到了这篇文章,但它在java 8之前已经很老了。


答案 1

就像@RealSkeptic注释中解释的那样,只能返回两个结果:true和false。这意味着您只能将数据拆分为两组。
你需要的是某种东西,这将允许你确定应该组合在一起的元素的一些常见结果。在你的例子中,这样的结果可能是小写的第一个字符(假设所有字符串都不是空的 - 至少有一个字符)。PredicateFunction

现在使用Collectors.groupingBy(函数),您可以将所有元素分组到单独的列表中,并将它们存储在Map中,其中键将是用于分组的常见结果(如第一个字符)。

所以你的代码可以看起来像

Function<String, Character> firstChar =  s -> Character.toLowerCase(s.charAt(0));

List<String> a = Arrays.asList("foo", "Abc", "bar", "baz", "aBc");
Map<Character, List<String>> collect = a.stream()
        .collect(Collectors.groupingBy(firstChar));

System.out.println(collect);

输出:

{a=[Abc, aBc], b=[bar, baz], f=[foo]}

答案 2

您可以使用 Collectors.groupingBy 将(分组)流转换为 >(该分组中的事物列表)。如果您不关心分组本身,请调用该地图以获取分区。values()Collection<List<String>>