吉森的技术小站 吉森的技术小站
首页
关于
  • 分类
  • 标签
  • 归档
GitHub (opens new window)

吉森

Fuel your ambition
首页
关于
  • 分类
  • 标签
  • 归档
GitHub (opens new window)
  • Java基础

    • 就从Java8开始吧(一)lambda表达式详解
    • 就从java8开始吧(二)lambda表达式和方法引用
    • 就从Java8开始吧(三)说一说Stream
    • 就从Java8开始吧(四)唠一唠Optional
    • 提高开发效率的奇技淫巧(一)lombok
    • maven概览
    • 就从Java8开始吧(五)新日期和时间API
    • 就从Java8开始吧(六)接口默认方法和静态方法
    • 提高开发效率的奇技(二)Intellij IDEA的进阶应用
    • 深入浅出java web(一):从根说起——servlet
    • 深入浅出java web (二):简述http协议
    • Java中equals()与==的区别详解
    • Java代理模式详解
    • Java中System.getProperty("user.dir")详解
    • Java格式工厂(一)——什么是二进制文件?
    • Java菜谱(二)——怎么求男学生的平均分?
      • 问题场景
      • 传统思路
      • Stream流式计算
      • 更复杂的场景
        • 复杂场景1:学生分属于不同班,计算每个班男同学的平均分
        • 复杂场景2:计算分数高于平均分的学生人数
    • Java菜谱(三)——常用数据结构转换及处理
    • Java菜谱(四)——怎么将10万条数据导出到excel?
    • Java菜谱(五)——怎么把字符串列表合并为一个字符串?
    • 函数式编程入门——拥抱函数式时代
    • Java Stream findFirst方法的空指针陷阱详解
  • Spring框架

  • 第三方库

  • Java
  • Java基础
吉森
2021-04-19
目录

Java菜谱(二)——怎么求男学生的平均分?

Stream函数式编程集合操作 0 人阅读

# 问题场景

今天的场景设计是这样的:

给定一批学生分数的数据,求出所有男学生的平均分数。 如果这个命题放在sql中,应该是送分题。在Java中去实现,可能也没有那么难。但是当场景不断复杂化,我们就需要一些技巧来解决这类问题了。

假设Student类的数据结构如下:

@Data
public class Student {
    /**
     * 学生ID
     */
    private String id;
    /**
     * 学生姓名
     */
    private String name;
    /**
     * 学生年龄
     */
    private Integer age;
    /**
     * 学生性别 0-女 1-男
     */
    private Integer gender;
    /**
     * 学生成绩
     */
    private Double score;
}
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23

# 传统思路

假设学生的数据是以List<Student>的形式给出的,让我们先来回顾一下传统思路是怎么解决这个问题的,由于

平均分数 = 总分 / 人数

因此,我们需要一个临时变量去记录总分,另一个临时变量去记录男学生的人数,然后我们遍历学生的列表,如果遍历到的学生为男学生,则总分加上当前学生的分数,人数加1,相关代码如下:

Double totalScore = 0.0;
int count = 0;
for (Student student : students) {
    // 男学生
    if (student.getGender() == 1) {
        totalScore += student.getScore();
        count++;
    }
}
Double average = totalScore / count;
System.out.println(average);
1
2
3
4
5
6
7
8
9
10
11

这样的思路属于命令式编程的范式,即我们一步一步告诉计算机先做什么再做什么,其好处是逻辑简单,容易理解和编写,也容易调试。但是这样的方式编程通常代码量巨大,并且很容易编写出执行效率低下的代码,处理复杂逻辑时更是容易丢掉代码的可读性。

# Stream流式计算

在Jdk8以后,Java引入了lambda表达式,使得Java可以更方便地使用函数式的风格编写程序。而同一版本中Stream的引入更是极大简化了集合的操作。

那么就让我们来看一下在Stream的帮助下如何解决上面的问题:

Double average = students.stream()
                .filter(s -> s.getGender() == 1)
                .collect(Collectors.averagingDouble(Student::getScore));

System.out.println(average);
1
2
3
4
5

首先通过列表的stream()方法将列表转为流,再通过filter方法对流中的元素进行过滤,最后通过collect方法对流中的元素进行归并,得到最终的结果。事实上,所有使用流的场景都遵循这三个步骤,即流的创建、流的转换以及流的归并。

上述流式计算的方式是一种函数式编程的风格,同时也是属于声明式编程的范式。相比于命令式编程,声明式编程更强调告诉计算机要做什么,而不是具体怎么做。每个步骤具体的实现方案由计算机内部自行实现。当然,这也依赖于Jdk内部提供的强大的api。

# 更复杂的场景

让我们把场景变得更复杂一些,来见识一样流式计算的威力。

# 复杂场景1:学生分属于不同班,计算每个班男同学的平均分

学生的类增加相应字段,改造为:

@Builder
@Data
public class Student {
    /**
     * 学生ID
     */
    private String id;
    /**
     * 学生姓名
     */
    private String name;
    /**
     * 学生年龄
     */
    private Integer age;
    /**
     * 学生性别 0-女 1-男
     */
    private Integer gender;
    /**
     * 学生成绩
     */
    private Double score;
    /**
     * 学生属于哪个班
     */
    private Integer classNumber;
}
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28

上述需求实现代码如下:

final Map<Integer, Double> averageMap = students.stream()
                .filter(s -> s.getGender() == 1)
                .collect(Collectors.groupingBy(Student::getClassNumber, 
                    Collectors.averagingDouble(Student::getScore)));
System.out.println(averageMap);
1
2
3
4
5

由于需要每个班的成绩,我们对学生按班级进行分组,使用的是Collectors工具类提供的groupingBy()方法。这个方法第一个参数是分类的依据,这里传的是Student::getClassNumber这个方法引用,即怎么根据学生对象获取到学生的班级。第二个参数传的是下游的收集器,即分组之后对每组元素做怎样的操作,这里和之前一样传的是对学生的成绩取平均分的操作。如果我们只对数据进行分组,不进行后续处理,第二个参数可以不传(重载方法)。

# 复杂场景2:计算分数高于平均分的学生人数

// 先求平均分
final Double average = students.stream()
    .collect(Collectors.averagingDouble(Student::getScore));

// 再求超过平均分的人数
final long count = students.stream()
    .filter(s -> s.getScore() > average)
    .count();
System.out.println(count);
1
2
3
4
5
6
7
8
9

这个需求想整合成一次流式操作比较困难,我们需要先获取班级的平均分,再去计算分数超过平均分的人数。需要注意的是,Stream对象是“一次性的”,当一次归并操作完成后,Stream就会被关闭,这时如果复用之前的对象就会抛出异常。

这里只举这两个例子,Stream还有很多方便的API,感兴趣的可以自行尝试。总结一下,使用Stream可以极大简化集合相关的操作,如果有相关的数据处理需求,可以尝试使用。

编辑 (opens new window)
#Stream#函数式编程#集合操作
上次更新: 2025/08/08, 17:31:58
Java格式工厂(一)——什么是二进制文件?
Java菜谱(三)——常用数据结构转换及处理

← Java格式工厂(一)——什么是二进制文件? Java菜谱(三)——常用数据结构转换及处理→

最近更新
01
怎么写好技术文章?
08-25
02
CommonJS与ES模块:新手完全指南
08-21
03
Java Stream findFirst方法的空指针陷阱详解
08-14
更多文章>
Theme by Vdoing | Copyright © 2024-2025 吉森 | MIT License | 吉ICP备17006653号
  • 跟随系统
  • 浅色模式
  • 深色模式
  • 阅读模式