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

Java streams,从2个列表中获取数据,java,lambda,java-8,java-stream,Java,Lambda,Java 8,Java Stream,我有以下两个列表,我正在尝试打印(通过Java流)每个课程的平均值和名称。 现在我的代码输出只给出了第一门课程的名称和总和,但我没有得到其他课程的名称和总和 如何修复我的代码?这是我的代码: courses.stream() .forEach(action -> System.out.println(action.getName()+" " +students.stream()

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

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

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));
    }};
List students=new LinkedList(){{
加(
新学生(“Moshe”,21,1,新LinkedList(){{
添加(新成绩(courses.stream().filter(c->c.getId()==30.findFirst().get(),67));
添加(新成绩(courses.stream().filter(c->c.getId()==32.findFirst().get(),89));
添加(新成绩(courses.stream().filter(c->c.getId()==35.findFirst().get(),67));
添加(新成绩(courses.stream().filter(c->c.getId()==37.findFirst().get(),89));
}})
);
加(
新学生(“Yossi”,26岁,2岁,新LinkedList(){{
添加(新成绩(courses.stream().filter(c->c.getId()==30.findFirst().get(),100));
添加(新成绩(courses.stream().filter(c->c.getId()==32.findFirst().get(),67));
添加(新成绩(courses.stream().filter(c->c.getId()==35.findFirst().get(),89));
}})
);
加(
新学生(“娜塔莎”,30岁,3岁,新链接列表(){{
添加(新成绩(courses.stream().filter(c->c.getId()==30.findFirst().get(),67));
添加(新成绩(courses.stream().filter(c->c.getId()==37.findFirst().get(),80));
}})
);
}};
列出课程=新建LinkedList(){{
增加(新课程(30,“数据结构”,3.5f));
增加(新课程(32,“几何”,6));
增加(新课程(35,“代数”,2.5f));
增加(新课程(37,“英语”,7));
}};

您的代码在
findFirst()之后调用的
get
方法中抛出一个
java.util.NoSuchElementException:No value present
。当
findFirst()
返回空的
可选值时,会发生这种情况

为什么代码中会出现这种情况?对于您的学生流中的每个学生,您可以创建一个成绩流,您可以根据课程名称进行筛选,以仅获取我们当前汇总的课程的成绩。如果所讨论的学生没有所讨论课程的分数,则会从流中筛选出所有分数,
findFirst()
返回一个空的
可选的

正如您所说,代码会打印第一个课程的结果,但是:

Data Structures 234.0
该代码适用于本课程,因为所有学生都已获得该代码,因此永远不会出现成绩的空流。下一门课是几何,因为娜塔莎还没有几何成绩。所以这里发生了例外

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

    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()
从这些等级流生成元素流。如果一个学生在一门课程中没有得到分数,那就意味着没有向流提供任何分数元素。现在我们可以把我们得到的所有分数映射成双倍并求和

输出是

Data Structures 234.0
Geometry 156.0
Algebra 156.0
English 169.0

顺便说一句,你的
课程
课程不应该有
等于
的方法吗?然后可以将
grade->grade.getCourse().getName().equals(course.getName())
简化为
grade->grade.getCourse().equals(course)
您的代码在
findFirst()之后调用的
get
方法中抛出
java.util.NoSuchElementException:不存在值。当
findFirst()
返回空的
可选值时,会发生这种情况

为什么代码中会出现这种情况?对于您的学生流中的每个学生,您可以创建一个成绩流,您可以根据课程名称进行筛选,以仅获取我们当前汇总的课程的成绩。如果所讨论的学生没有所讨论课程的分数,则会从流中筛选出所有分数,
findFirst()
返回一个空的
可选的

正如您所说,代码会打印第一个课程的结果,但是:

Data Structures 234.0
该代码适用于本课程,因为所有学生都已获得该代码,因此永远不会出现成绩的空流。下一门课是几何,因为娜塔莎还没有几何成绩。所以这里发生了例外

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

    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()
从这些等级流生成元素流。如果一个学生在一门课程中没有得到分数,那就意味着没有向流提供任何分数元素。现在我们可以把我们得到的所有分数映射成双倍并求和

输出是

Data Structures 234.0
Geometry 156.0
Algebra 156.0
English 169.0

顺便说一句,你的
课程
课程不应该有
等于
的方法吗?然后可以将
grade->grade.getCourse().getName().equals(course.getName())
简化为
grade->grade.getCourse().equals(course)
,因为成绩有一个指向课程的链接,所以您只能使用一个流。 把所有学生的分数都记下来,然后放在地图上。 其中key-课程名称,值为sum

    Map<String, Integer> map = students.stream()
            .flatMap(s -> s.getGrades().stream())
            .collect(
                    Collectors.toMap(g -> g.getCourse().getName(),
                            Grade::getValue, 
                           (oldValue, newValue) -> oldValue + newValue));
    map.entrySet().forEach(es -> System.out.println(es.getKey() + " " + es.getValue()));
Map Map=students.stream()
.flatMap(s->s.getGrades().stream())
.收集(
Collectors.toMap(g->g.getCourse().getName(),
等级::getValue,
(oldValue,newValue)->oldValue+newValue);
map.entrySet().forEach(es->System.out.println(es.getKey()+“”+es.getValue());
因为等级有一个链接