代码看起来一点也不优雅,试试Stream

简介: 代码看起来一点也不优雅,试试Stream

前言

JDK的发行版本都已经衍生至19了,这个从8就引入的Stream流应当是属于Java程序员基操了。然而最近面试遇到开发经验3年的工程师,对它似乎不是很熟悉,让我大吃一惊。本文通过常用的一些简单的例子把它以最小的时间成本给大家说明白。

一、楔子

在使用集合的时候迭代往往是使用的最多的,对比是否使用stream的代码实现,

public int calcSum(List<Integer> list) {
        int sum = 0;
        for (int i = 0; i < list.size(); i++) {
            sum += list.get(i);
        }
        return sum;
    }
    public int calcSumLambda(List<Integer> list) {
        return list.stream().mapToInt(x -> x).sum();
    }

第一次看到这样的写法时,可能会认为这样的代码可读性不高,但当你熟悉之后,你会改变对它的看法。

二、如何创建流

想要使用Stream,首先需要创建一个流,最常见的是直接调用集合的java.util.Collection#stream方法

private void createByCollection() {
        List<Integer> list = new ArrayList<>();
        Stream<Integer> stream = list.stream();
    }

当然通过数组同样能够创建

private void createByArrays() {
        Integer[] array1 = {1, 2, 3};
        Integer[] array2 = new Integer[] {1, 2, 3};
        Stream<Integer> stream1 = Stream.of(array1);
        Stream<Integer> stream2 = Arrays.stream(array1);
    }

三、流操作的分类

Stream流操作共分为两个大类:惰性求值、及早求值

/**
     * 通过Stream流过滤元素返回新的集合
     * 
     * @param list 待过滤的集合
     * @return 新的集合
     */
    private List<Integer> filterByStream(List<Integer> list) {
        return list.stream().filter(number -> number > 1).collect(Collectors.toList());
    }

Stream操作时,先调用了filter方法传入了一个Lambda表达式代表过滤规则,后调用了collect方法表示将流转换为List集合。

我们不需要去记哪些方法是惰性求值,如果方法的返回值是Stream那么它代表的就是惰性求值。如果返回另外一个值或空,那么它代表的就是及早求值。

这些流操作定义之后,在程序中是怎么调用的定义的lambda表达式的?

例如在java.util.stream.ReferencePipeline抽象类中有对Stream接口collect的实现,方法由final修饰,不再支持重写。

@Override
    @SuppressWarnings("unchecked")
    public final <R, A> R collect(Collector<? super P_OUT, A, R> collector) {
        A container;
        if (isParallel()
                && (collector.characteristics().contains(Collector.Characteristics.CONCURRENT))
                && (!isOrdered() || collector.characteristics().contains(Collector.Characteristics.UNORDERED))) {
            container = collector.supplier().get();
            BiConsumer<A, ? super P_OUT> accumulator = collector.accumulator();
            forEach(u -> accumulator.accept(container, u));
        }
        else {
            container = evaluate(ReduceOps.makeRef(collector));
        }
        return collector.characteristics().contains(Collector.Characteristics.IDENTITY_FINISH)
               ? (R) container
               : collector.finisher().apply(container);
    }

四、常用基操

map

映射,x->y,转换数据类型

/**
     * 通过Stream map操作将小写的字符串集合转换为大写
     * 
     * @param list 小写字符串集合
     * @return 大写字符串集合
     */
    public List<String> toUpperByStreamMap(List<String> list) {
        // return list.stream().map(String::toUpperCase).collect(Collectors.toList());
        return list.stream().map(string -> string.toUpperCase()).collect(Collectors.toList());
    }

filter

过滤,“排除不符合某个条件的元素”,也就是返回true的时候保留,返回false排除

/**
     * 通过Stream filter筛选出分数大于60分的学生集合
     * 
     * @param students 待过滤的学生集合
     * @return 分数大于60分的学生集合
     */
    public List<Student> fetchPassedStudentsByStreamFilter(List<Student> students) {
        return students.stream().filter(student -> student.getScore().compareTo(60) >= 0).collect(Collectors.toList());
    }

sorted

排序,

/**
     * 学生按分数升序排列
     * 
     * @param students 学生集合
     * @return 排序后的学生集合
     */
    private List<Student> sortedByStreamSorted(List<Student> students) {
        return students.stream().sorted(Comparator.comparing(Student::getScore)).collect(Collectors.toList());
    }

如果要降序(大-->小),仅需再调用reversed方法Comparator.comparing(Student::getScore).reversed())这就是声明式编程,你只管叫它做什么,而不像命令式编程叫它如何做。

reduce

对于reduce操作,不建议在现实中使用。

如果你有累加、求最大值、最小值的需求,Stream封装了更简单的方法。

/**
     * T reduce(T identity, BinaryOperator<T> accumulator);
     * 赋初始值为1,对集合中的元素进行累加
     * 
     * @param numbers 集合元素
     * @return 累加结果
     */
    private Integer calcTotal2(List<Integer> numbers) {
        return numbers.stream().reduce(1, (total, number) -> total + number);
    }

min || max

顾名思义,求取集合中的最小值和最大值。

Java8对Comparator接口提供了新的静态方法comparing,这个方法返回Comparator对象,以前我们需要手动实现compare比较,现在我们只需要调用Comparator.comparing静态方法即可。

/**
     * 通过Stream min计算集合中的最小值
     * 
     * @param numbers 集合
     * @return 最小值
     */
    private Integer minByStreamMin(List<Integer> numbers) {
        return numbers.stream().min(Comparator.comparingInt(Integer::intValue)).get();
    }
    /**
     * 通过Stream min计算学生集合中的最低成绩
     * 
     * @param students 学生集合
     * @return 最低成绩
     */
    private Double minScoreByStreamMin(List<Student> students) {
        Student minScoreStudent = students.stream().min(Comparator.comparing(Student::getScore)).get();
        return minScoreStudent.getScore();
    }

summaryStatistics

求和操作也是常用的操作,利用reduce会让代码晦涩难懂,特别是复杂的对象类型。

/**
     * 学生类型的集合常用计算
     * 
     * @param students 学生
     */
    private void calc(List<Student> students) {
        DoubleSummaryStatistics summaryStatistics =
            students.stream().mapToDouble(Student::getScore).summaryStatistics();
        System.out.println("平均分:" + summaryStatistics.getAverage());
        System.out.println("总分:" + summaryStatistics.getSum());
        System.out.println("最高分:" + summaryStatistics.getMax());
        System.out.println("最低分:" + summaryStatistics.getMin());
    }

Collectors

前面的大部分操作都是以collect(Collectors.toList())结尾,看多了自然也大概猜得到它是将流转换为集合对象。最大的功劳当属Java8新提供的类——Collectors收集器。

示例给出比较常见的List和Map的转换,

/**
     * 将学生类型的集合转换为只包含名字的集合
     * 
     * @param students 学生集合
     * @return 学生姓名集合
     */
    private List<String> translateNames(List<Student> students) {
        return students.stream().map(Student::getStudentName).collect(Collectors.toList());
    }
    /**
     * 将学生类型的集合转换为Map类型,key=学号,value=学生
     * 
     * @param students 学生集合
     * @return 学生Map
     */
    private Map<Long, Student> translateStudentMap(List<Student> students) {
        return students.stream().collect(Collectors.toMap(Student::getStudentNumber, student -> student));
    }

小结

以上内容没有多高深,主要为了让大家知道不同情景下该使用哪种API,如果有同学都不熟悉这些,还怎么写出优雅的代码呢,面试通过也成问题,像我就喜欢编程功底扎实的(手动狗头)。

相关文章
|
8月前
|
传感器 JSON Dart
Dart笔记:stream_channel 包用法
Dart笔记:stream_channel 包用法
194 0
|
8月前
|
Go
Go命令行解析神器入门 - 10分钟上手flag包
Go命令行解析神器入门 - 10分钟上手flag包
214 0
5万字长文:Stream和Lambda表达式最佳实践-附PDF下载(一)
5万字长文:Stream和Lambda表达式最佳实践-附PDF下载(一)
5万字长文:Stream和Lambda表达式最佳实践-附PDF下载(一)
|
存储 机器学习/深度学习 网络协议
逐行解释webserver源码(window版)
逐行解释webserver源码(window版)
|
开发框架 .NET 测试技术
如何使用go做命令行flag解析
如何使用go做命令行flag解析
112 0
|
并行计算 Java 测试技术
5万字长文:Stream和Lambda表达式最佳实践-附PDF下载(三)
5万字长文:Stream和Lambda表达式最佳实践-附PDF下载(三)
5万字长文:Stream和Lambda表达式最佳实践-附PDF下载(三)
|
Go uml Python
读 Go 源码,可以试试这个工具
读 Go 源码,可以试试这个工具
363 0
读 Go 源码,可以试试这个工具
|
Java API 数据处理
用Stream来优化老代码,就是爽
用Stream来优化老代码,就是爽
169 0
|
JSON vr&ar 图形学
【100个 Unity小知识点】 | Unity读取txt或者Json文件,包括 换行符 的方法
Unity 小科普 老规矩,先介绍一下 Unity 的科普小知识: Unity是 实时3D互动内容创作和运营平台 。 包括游戏开发、美术、建筑、汽车设计、影视在内的所有创作者,借助 Unity 将创意变成现实。 Unity 平台提供一整套完善的软件解决方案,可用于创作、运营和变现任何实时互动的2D和3D内容,支持平台包括手机、平板电脑、PC、游戏主机、增强现实和虚拟现实设备。 也可以简单把 Unity 理解为一个游戏引擎,可以用来专业制作游戏!
【100个 Unity小知识点】 | Unity读取txt或者Json文件,包括 换行符 的方法