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

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

给定一批学生分数的数据,求出所有男学生的平均分数。

如果这个命题放在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;
}

传统思路

假设学生的数据是以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);

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

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);

首先通过列表的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;
}

上述需求实现代码如下:

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);

由于需要每个班的成绩,我们对学生按班级进行分组,使用的是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);

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

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

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值