网站首页 > 精选教程 正文
今天的场景设计是这样的:
给定一批学生分数的数据,求出所有男学生的平均分数。
如果这个命题放在sql中,应该是送分题。在Java中去实现,可能也没有那么难。但是当场景不断复杂化,我们就需要一些技巧来解决这类问题了。现在我们就来探讨一下如何通过Stream流式编程解决这一类问题。
假设Student类的数据结构如下:
@Data
public class Student {
/**
* 学生ID
*/
private String id;
/**
* 学生姓名
*/
private String name;
/**
* 学生年龄
*/
private Integer age;
/**
* 学生性别 0-女 1-男
*/
private Integer gender;
/**
* 学生成绩
*/
private Double score;
}
传统思路
假设学生的数据是以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;
这样的思路属于命令式编程的范式,即我们一步一步告诉计算机先做什么再做什么,其好处是逻辑简单,容易理解和编写,也容易调试。但是这样的方式编程通常代码量巨大,并且很容易编写出执行效率低下的代码,处理复杂逻辑时更是容易丢掉代码的可读性。
Stream流式计算
在Jdk8以后,Java引入了lambda表达式和Stream。Lambda表达式使得Java可以更方便地使用函数式的风格编写程序,Stream的引入更是极大简化了集合的操作。
那么就让我们来看一下在Stream的帮助下如何解决上面的问题:
Double average = students.stream()
.filter(s -> s.getGender() == 1)
.collect(Collectors.averagingDouble(Student::getScore));
上述代码看上去很简短,但实际包含了三个步骤:
- 首先通过列表的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;
}
上述需求实现代码如下:
Map<Integer, Double> averageMap = students.stream()
.filter(s -> s.getGender() == 1)
.collect(Collectors.groupingBy(Student::getClassNumber,
Collectors.averagingDouble(Student::getScore)));
由于需要每个班的成绩,我们对学生按班级进行分组,使用的是Collectors工具类提供的groupingBy()方法。这个方法第一个参数是分类的依据,这里传的是Student::getClassNumber这个方法引用,即怎么根据学生对象获取到学生的班级。第二个参数传的是下游的收集器,即分组之后对每组元素做怎样的操作,这里和之前一样传的是对学生的成绩取平均分的操作。如果我们只对数据进行分组,不进行后续处理,第二个参数可以不传(重载方法)。
复杂场景2:计算分数高于平均分的学生人数
// 先求平均分
Double average = students.stream()
.collect(Collectors.averagingDouble(Student::getScore));
// 再求超过平均分的人数
long count = students.stream()
.filter(s -> s.getScore() > average)
.count();
这个需求想整合成一次流式操作比较困难,因此我们可以先获取班级的平均分,再去计算分数超过平均分的人数。
需要注意的是,Stream对象是“一次性的”,当一次归并操作完成后,Stream就会被关闭,这时如果复用之前的对象就会抛出异常。
这里只举这两个例子,Stream还有很多方便的API,感兴趣的可以自行尝试。总结一下,使用Stream可以极大简化集合相关的操作,如果有相关的数据处理需求,可以尝试使用。
欢迎各位留言讨论,如果您觉得有帮助,还请多多点赞、转发支持~
猜你喜欢
- 2024-11-10 为什么28岁不能学java? 28岁开始学编程晚吗
- 2024-11-10 gc的年龄信息存储在什么地方? 年龄信息用什么数据类型存储
- 2024-11-10 JVM基础学习 jvm入门到精通
- 2024-11-10 Java-对象到底占多少个字节?计算规则是什么?
- 2024-11-10 浅谈JAVA中的基本变量 java常用变量
- 2024-11-10 大数据必学Java基础(十三):基本数据类型的转换
- 2024-11-10 Java码农35岁之后只能送外卖? 我的世界java
- 2024-11-10 这些Java基础知识,诸佬们都还记得嘛(学习,复习,面试都可)
- 2024-11-10 2021-01-03:java中,描述一下什么情况下,...
- 2024-11-10 零基础学Java之数据类型与变量(上集)
你 发表评论:
欢迎- 最近发表
- 标签列表
-
- nginx反向代理 (57)
- nginx日志 (56)
- nginx限制ip访问 (62)
- mac安装nginx (55)
- java和mysql (59)
- java中final (62)
- win10安装java (72)
- java启动参数 (64)
- java链表反转 (64)
- 字符串反转java (72)
- java逻辑运算符 (59)
- java 请求url (65)
- java信号量 (57)
- java定义枚举 (59)
- java字符串压缩 (56)
- java中的反射 (59)
- java 三维数组 (55)
- java插入排序 (68)
- java线程的状态 (62)
- java异步调用 (55)
- java中的异常处理 (62)
- java锁机制 (54)
- java静态内部类 (55)
- java怎么添加图片 (60)
- java 权限框架 (55)
本文暂时没有评论,来添加一个吧(●'◡'●)