Java流,从2个列表中获取数据

标签 java lambda java-8 java-stream

我有以下 2 个列表,我正在尝试打印(通过 Java Stream)每门类(class)的平均值和名称。 我的代码的输出现在只给我第一门类(class)的名称和总和,但我没有得到其他类(class)

如何修复我的代码?这是我的代码:

courses.stream()
       .forEach(action -> System.out.println(action.getName()+" "
                    +students.stream()
                             .mapToDouble(f->f.getGrades().stream()
                                     .filter(predicate->predicate.getCourse()
                                             .getName()
                                             .equals(action.getName()))
                                     .findFirst()
                                     .get()
                                     .getValue())
                             .sum() ));`

这是列表:

     List<Student> students = new LinkedList<Student>() {{
        add(
                new Student("Moshe", 21, 1, new LinkedList<Grade>() {{
                    add(new Grade(courses.stream().filter(c -> c.getId() == 30).findFirst().get(), 67));
                    add(new Grade(courses.stream().filter(c -> c.getId() == 32).findFirst().get(), 89));
                    add(new Grade(courses.stream().filter(c -> c.getId() == 35).findFirst().get(), 67));
                    add(new Grade(courses.stream().filter(c -> c.getId() == 37).findFirst().get(), 89));
                }})
        );
        add(
                new Student("Yossi", 26, 2, new LinkedList<Grade>() {{
                    add(new Grade(courses.stream().filter(c -> c.getId() == 30).findFirst().get(), 100));
                    add(new Grade(courses.stream().filter(c -> c.getId() == 32).findFirst().get(), 67));
                    add(new Grade(courses.stream().filter(c -> c.getId() == 35).findFirst().get(), 89));
                }})
        );
        add(
                new Student("Natasha", 30, 3, new LinkedList<Grade>() {{
                    add(new Grade(courses.stream().filter(c -> c.getId() == 30).findFirst().get(), 67));
                    add(new Grade(courses.stream().filter(c -> c.getId() == 37).findFirst().get(), 80));

                }})
        );
    }};

   List<Course> courses = new LinkedList<Course>() {{
        add(new Course(30, "Data Structures", 3.5f));
        add(new Course(32, "Geometry", 6));
        add(new Course(35, "Algebra", 2.5f));
        add(new Course(37, "English", 7));
    }};

最佳答案

您的代码从 findFirst() 之后调用的 get 方法抛出 java.util.NoSuchElementException: No value present。当 findFirst() 返回空的 Optional 时,就会发生这种情况。

为什么你的代码中会发生这种情况?对于学生流中的每个学生,您都会创建一个成绩流,并按类(class)名称进行过滤,以仅获取我们当前汇总的类(class)的成绩。如果相关学生没有相关类(class)的成绩,则所有成绩都会从流中过滤掉,并且 findFirst() 返回空的 Optional

正如您所说,代码确实打印了第一门类(class)的结果:

Data Structures 234.0

该代码适用于本类(class),因为所有学生都已获得它,因此永远不会出现空的成绩流。对于下一门类(class),几何,将会有,因为娜塔莎尚未获得几何成绩。所以这里发生了异常。

解决方案与您的代码相差不远:

    courses.stream()
            .forEach(course -> System.out.println(course.getName() + ' '
                    + students.stream()
                        .flatMap(student -> student.getGrades()
                                .stream()
                                .filter(grade -> grade.getCourse()
                                                    .getName()
                                                    .equals(course.getName())))
                        .mapToDouble(Grade::getValue)
                        .sum()));

我引入的新元素是学生流中的 flatMap() 操作。 flatMap() 的参数生成一个成绩流,可能为空。 flatMap() 从这些等级流中生成元素流。如果学生没有在类(class)中获得成绩,则仅意味着没有向流提供成绩元素。现在我们可以将我们得到的所有成绩映射到 double 并对它们求和。

输出为

Data Structures 234.0
Geometry 156.0
Algebra 156.0
English 169.0

顺便说一句,您的 Course 类不应该有一个 equals 方法吗?然后 grade ->grade.getCourse().getName().equals(course.getName()) 可以简化为 grade ->grade.getCourse().equals(course).

关于Java流,从2个列表中获取数据,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/44475615/

相关文章:

java - 安全 Java 应用程序数据库

java - 使用 java API 搜索 Hbase 记录?

java - JSF:如何写入数据库?

c# - 如何简化Where(e => e.prop1.contains() || e.prop2.contains() || ...)中的重复OR条件

Java 8 : map. 合并时间复杂度

java - 为什么我会收到未经检查的类型转换警告?

c# - C# 或 Java 有 lambda 的原因是什么?

使用显式定义的谓词时出现 C# 错误

java - 使用 Stream API 创建对象初始化循环

java - 使用 java 流将 Stream<Object[]> 转换为 List<Object>