我有以下 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/