Java-Stream - 使用过滤器和 forEach 填充集合

new*_*ner 2 java collections side-effects java-stream java-16

我有一个Set<Person>,我需要过滤内容,然后填充给定集中的另一组。

我对流还很陌生。

我能想到的第一个解决方案是链接stream().filter().foreach()

解决方案一:

private Set<Person> mapPerson(final Set<Student> student) {
        final Set<Person> person = new HashSet<>();
        student.stream()
            .filter(student1 -> Objects.nonNull(student.getAge()))
            .forEach(student1 -> person.add(Person.builder()
                .id(student1.getId())
                .status(student1.getStatus()))
                .build()));

        return person;
    }
Run Code Online (Sandbox Code Playgroud)

但我不确定这是否可以通过链接来完成stream().filter().map()

将不胜感激任何帮助,如果可以两种方式进行,哪种方式是首选方式?

Ale*_*nko 5

为此,您不需要手动创建一个集合然后对其进行变异。您可以获得一个集合作为流管道执行的结果。

可以通过链接stream().filter().map()以某种方式完成吗?

首先通过应用操作将学生流转换Stream<Student>为人员对象流 ,该操作需要一个将流元素转换为所需类型的另一个对象的函数。因此,您需要将创建内部使用的对象的逻辑放入.Stream<Person>map()PersonforEach()map()

然后应用collect作为参数传递Collectors.toSet()

private Set<Person> mapPerson(final Set<Student> students) {

  return students.stream()
          .filter(student -> Objects.nonNull(student.getAge()))
          .map(student -> Person.builder()
                          .id(student.getId())
                          .status(student.getStatus())
                          .build())
          .collect(Collectors.toSet());
}
Run Code Online (Sandbox Code Playgroud)

如果两种方法都可以的话,哪种方法是首选?

文档不鼓励您使用的方法看看下面的引用(强调是我的)。

作为如何将 不适当地使用副作用的流管道转换为不使用副作用的流管道的示例,以下代码在字符串流中搜索与给定正则表达式匹配的字符串,并将匹配项放入列表中。

 ArrayList<String> results = new ArrayList<>();
 stream.filter(s -> pattern.matcher(s).matches())
       .forEach(s -> results.add(s));  // Unnecessary use of side-effects!
Run Code Online (Sandbox Code Playgroud)

... forEach()可以简单地替换为更安全、更高效且更适合并行化的归约操作:

 List<String> results =
     stream.filter(s -> pattern.matcher(s).matches())
           .toList();  // No side-effects!
Run Code Online (Sandbox Code Playgroud)

您不应该使用forEach(以及peek),它只能通过在有其他方法来实现相同结果的情况下引起副作用来操作。

collect()在这种情况下,您可以使用(如上所示)或使用普通循环来完成for

  • @newLearner 在您的代码中,集合是通过“forEach()”的副作用来填充的。如果可以通过其他方式使用,则不鼓励使用。由于在这种情况下,还有其他方法可以实现这一点,因此没有理由诉诸“forEach”。查看[文档](https://docs.oracle.com/en/java/javase/17/docs/api/java.base/java/util/stream/package-summary.html#SideEffects)。 (2认同)
  • @newLearner 这样的话就不会发生悲剧了。您的初始代码将正确执行。但正如我所设置的,这是一种不鼓励的做法,而且它并不能真正给你带来任何东西,相反,它使代码更加冗长且缺乏表达力,因为你需要单独声明集合。 (2认同)