如何使用Java8通过谓词对列表进行分区?


问题内容

我有一个清单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之前已经很老了。


问题答案:

就@RealSkeptic中解释的那样,注释Predicate只能返回两个结果:true和false。这意味着您将只能将数据分为两组。
您需要的是某种Function可以让您确定应该分组在一起的元素的常见结果。在您的情况下,这样的结果可能是小写的第一个字符(假设所有字符串都不为空-至少有一个字符)。

现在,Collectors.groupingBy(function)您可以将所有元素分组到单独的列表中,并将它们存储在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]}