Java-函数式编程(三)流(Stream)

简介: Java-函数式编程(三)流(Stream)

从外部迭代到内部迭代



什么是外部迭代和内部迭代呢?


个人认为,外和内是相对集合代码而言。


如果迭代的业务执行在应用代码中,称之为外部迭代。


反之,迭代的业务执行在集合代码中,称为内部迭代(函数式编程)。


语言描述可能有点抽象,下面看实例。


1. 外部迭代


调用itrator方法,产生一个新的Iterator对象,进而控制整个迭代过程。


for (Student student:list){
    if (student.getAge()>18){
        result++;
    }
}


我们都知道,for其实底层使用的迭代器:


Iterator<Student> iterator = list.iterator();
while (iterator.hasNext()){
    Student student = iterator.next();
    if (student.getAge()>18){
        result++;
    }
}


上面的迭代方法就是外部迭代。


外部迭代缺点:


  1. 很难抽象出复杂操作


  1. 本质上讲是串行化操作。


2. 内部迭代


返回内部迭代中的响应接口:Stream


long count = list.stream().filter(student -> student.getAge() > 18).count();



整个过程被分解为:过滤和计数。


要注意:返回的Stream对象不是一个新集合,而是创建新集合的配方。


2.1 惰性求值和及早求值


像filter这样值描述Stream,最终不产生新集合的方法叫做惰性求值



像count这样最终会从Stream产生值的方法叫做及早求值


判断一个操作是惰性操作还是及早求值,只需看它的返回值。如果返回值是Stream,那么是惰性求值;如果返回值是另一个值或者为空,那就是及早求值。这些操作的理想方式就是形成一个惰性求值的链,最后用一个及早求值的操作返回想要的结果。


整个过程跟建造者模式很像,使用一系列的操作后最后调用build方法才返回真正想要的对象。设计模式快速学习(四)建造者模式


那这个过程有什么好处呢:可以在集合类上级联多种操作,但迭代只需要进行一次。


3. 常用操作


3.1 collect(toList())   及早求值


collect(toList())方法由Stream里的值生成一个列表,是一个及早求值操作。


List<String> collect = Stream.of("a", "b", "c").collect(Collectors.toList());

Stream.of("a", "b", "c")首先由列表生成一个Stream对象,然后collect(Collectors.toList())生成List对象。


3.2 map


map可以将一种类型的值转换成另一种类型。


List<String> streamMap = Stream.of("a", "b", "c").map(String -> String.toUpperCase()).collect(Collectors.toList());


map(String -> String.toUpperCase())将返回所有字母的大写字母的Stream对象,collect(Collectors.toList())返回List。


3.3 filter过滤器


遍历并检查其中的元素时,可用filter


List<String> collect1 = Stream.of("a", "ab", "abc")
        .filter(value -> value.contains("b"))
        .collect(Collectors.toList());

3.4 flatMap


如果有一个包含了多个集合的对象希望得到所有数字的集合,我们可以用flatMap

List<Integer> collect2 = Stream.of(asList(1, 2), asList(3, 4))
        .flatMap(Collection::stream)
        .collect(Collectors.toList());


Stream.of(asList(1, 2), asList(3, 4))将每个集合转换成Stream对象,然后.flatMap处理成新的Stream对象。


3.5 max和min


看名字就知道,最大值和最小值。


Student student1 = list.stream()
        .min(Comparator.comparing(student -> student.getAge()))
        .get();


java8提供了一个Comparator静态方法,可以借助它实现一个方便的比较器。其中Comparator.comparing(student -> student.getAge()可以换成Comparator.comparing(Student::getAge)成为更纯粹的lambda。max同理。


3.6 reduce


reduce操作可以实现从一组值中生成一个值,在上述例子中用到的count、min、max方法事实上都是reduce操作。


Integer reduce = Stream.of(1, 2, 3).reduce(0, (acc, element) -> acc + element);
System.out.println(reduce);
6


上面的例子使用reduce求和,0表示起点,acc表示累加器,保存着当前累加结果(每一步都将stream中的元素累加至acc),element是当前元素。


4. 操作整合


  1. collect(toList())方法由Stream里的值生成一个列表
  2. map可以将一种类型的值转换成另一种类型。
  3. 遍历并检查其中的元素时,可用filter
  4. 如果有一个包含了多个集合的对象希望得到所有数字的集合,我们可以用flatMap
  5. max和min
  6. reduce(不常用)


5. 链式操作实战


List<Student> students = new ArrayList<>();
students.add(new Student("Fant.J",18));
students.add(new Student("小明",19));
students.add(new Student("小王",20));
students.add(new Student("小李",22));
List<Class> classList = new ArrayList<>();
classList.add(new Class(students,"1601"));
classList.add(new Class(students,"1602"));
static class Student{
        private String name;
        private Integer age;
        getter and setter ...and construct ....
    }
    static class Class{
        private List<Student> students;
        private String className;
        getter and setter ...and construct ....
    }


这是我们的数据和关系--班级和学生,现在我想要找名字以小开头的学生,用stream链式操作:

List<String> list= students.stream()
                            .filter(student -> student.getAge() > 18)
                            .map(Student::getName)
                            .collect(Collectors.toList());
[小明, 小王, 小李]


这是一个简单的students对象的Stream的链式操作实现,那如果我想要在许多个class中查找年龄大于18的对象呢?


6. 实战提升


在许多个class中查找年龄大于18的名字并返回集合。


原始代码:


List<String> nameList = new ArrayList<>();
        for (Class c:classList){
            for (Student student:c.getStudents()){
                if (student.getAge()>18){
                    String name = student.getName();
                    nameList.add(name);
                }
            }
        }
        System.out.println(nameList);


链式流代码:


如果让你去写,你可能会classList.stream().forEach(aClass -> aClass.getStudents().stream())....去实现?


我刚开始就是这样无脑干的,后来我缓过神来,想起foreach是一个及早求值操作,而且返回值是void,这样的开头就注定了没有结果,然后仔细想想,flatMap不是用来处理不是一个集合的流吗,好了,就有了下面的代码。


List<String> collect = classList.stream()
        .flatMap(aclass -> aclass.getStudents().stream())
        .filter(student -> student.getAge() > 18)
        .map(Student::getName)
        .collect(toList());


原始代码和流链式调用相比,有以下缺点


  1. 代码可读性差,隐匿了真正的业务逻辑
  2. 需要设置无关变量来保存中间结果
  3. 效率低,每一步都及早求值生成新集合
  4. 难于并行化处理


7. 高阶函数及注意事项


高阶函数是指接受另外一个函数作为参数,或返回一个函数的函数。如果函数的函数里包含接口或返回一个接口,那么该函数就是高阶函数。


Stream接口中几乎所有的函数都是高阶函数。比如:Comparing  接受一个函数作为参数,然后返回Comparator接口。


Student student = list.stream().max(Comparator.comparing(Student::getAge)).get();
public interface Comparator<T> {}


foreach方法也是高阶函数:

void forEach(Consumer<? super T> action);
public interface Consumer<T> {}


除了以上还有一些类似功能的高阶函数外,不建议将lambda表达式传给Stream上的高阶函数,因为大部分的高阶函数这样用会有一些副作用:给变量赋值。局部变量给成员变量赋值,导致很难察觉。


这里拿ActionEvent举例子:

image.png


关注后端技术精选,每天推送优质好文


目录
相关文章
|
22天前
|
安全 Java API
告别繁琐编码,拥抱Java 8新特性:Stream API与Optional类助你高效编程,成就卓越开发者!
【8月更文挑战第29天】Java 8为开发者引入了多项新特性,其中Stream API和Optional类尤其值得关注。Stream API对集合操作进行了高级抽象,支持声明式的数据处理,避免了显式循环代码的编写;而Optional类则作为非空值的容器,有效减少了空指针异常的风险。通过几个实战示例,我们展示了如何利用Stream API进行过滤与转换操作,以及如何借助Optional类安全地处理可能为null的数据,从而使代码更加简洁和健壮。
56 0
|
26天前
|
Java 程序员 API
解锁Java新纪元:Lambda表达式——让你的代码秒变高大上,函数式编程不再是梦!
【8月更文挑战第25天】Java 8 推出了革命性的 Lambda 表达式特性,这是一种匿名函数形式,支持任意数量参数及返回值,简化了代码编写。其基本语法为 `(parameters) -&gt; expression` 或 `(parameters) -&gt; { statements; }`。例如,遍历集合可从使用匿名内部类变为简洁的 `names.forEach(name -&gt; System.out.println(name))`。
36 0
|
9天前
|
Java API C++
Java 8 Stream Api 中的 peek 操作
本文介绍了Java中`Stream`的`peek`操作,该操作通过`Consumer&lt;T&gt;`函数消费流中的每个元素,但不改变元素类型。文章详细解释了`Consumer&lt;T&gt;`接口及其使用场景,并通过示例代码展示了`peek`操作的应用。此外,还对比了`peek`与`map`的区别,帮助读者更好地理解这两种操作的不同用途。作者为码农小胖哥,原文发布于稀土掘金。
Java 8 Stream Api 中的 peek 操作
|
9天前
|
Java C# Swift
Java Stream中peek和map不为人知的秘密
本文通过一个Java Stream中的示例,探讨了`peek`方法在流式处理中的应用及其潜在问题。首先介绍了`peek`的基本定义与使用,并通过代码展示了其如何在流中对每个元素进行操作而不返回结果。接着讨论了`peek`作为中间操作的懒执行特性,强调了如果没有终端操作则不会执行的问题。文章指出,在某些情况下使用`peek`可能比`map`更简洁,但也需注意其懒执行带来的影响。
Java Stream中peek和map不为人知的秘密
|
21天前
|
Java API
Java 8新特性:Lambda表达式与Stream API的深度解析
【7月更文挑战第61天】本文将深入探讨Java 8中的两个重要特性:Lambda表达式和Stream API。我们将首先介绍Lambda表达式的基本概念和语法,然后详细解析Stream API的使用和优势。最后,我们将通过实例代码演示如何结合使用Lambda表达式和Stream API,以提高Java编程的效率和可读性。
|
19天前
|
Java
盘点java8 stream中隐藏的函数式接口
`shigen`是一位坚持更新文章的博客作者,记录成长历程,分享认知见解,留住感动瞬间。本文介绍了函数式接口的概念及其在Java中的应用,包括`Comparator`、`Runnable`、`Callable`等常见接口,并详细讲解了`Function`、`Predicate`、`Consumer`、`Supplier`和`Comparator`等函数式接口的使用方法及应用场景,展示了如何利用这些接口简化代码并提高编程效率。**个人IP:shigen**,与shigen一起,每天进步一点点!
29 0
盘点java8 stream中隐藏的函数式接口
|
21天前
|
并行计算 Java 大数据
Java函数式编程:一场编程范式的革命,让你的代码焕发新生!
【8月更文挑战第30天】Java函数式编程是一种基于数学函数理论的编程范式,强调数据处理的不可变性和纯函数使用,通过将函数视为第一类对象,实现更简洁、易读的代码结构,在数据流处理与并行计算中尤为突出。与命令式编程关注执行步骤不同,函数式编程侧重描述计算目标而非具体操作流程,减少了状态变化,使代码更清晰易维护。在Java中,函数式编程通过降低副作用和状态依赖简化了复杂度,并提高了代码质量和测试性,尤其是在Java 8的Stream API中得到了充分体现,能够自动优化多核处理器上的并行处理性能。
32 2
|
1月前
|
分布式计算 Java API
Java 8带来了流处理与函数式编程等新特性,极大提升了开发效率
Java 8带来了流处理与函数式编程等新特性,极大提升了开发效率。流处理采用声明式编程模型,通过filter、map等操作简化数据集处理,提高代码可读性。Lambda表达式支持轻量级函数定义,配合Predicate、Function等接口,使函数式编程无缝融入Java。此外,Optional类及新日期时间API等增强功能,让开发者能更优雅地处理潜在错误,编写出更健壮的应用程序。
25 1
|
1月前
|
Java API 开发者
|
1月前
|
存储 算法 Oracle
19 Java8概述(Java8概述+lambda表达式+函数式接口+方法引用+Stream+新时间API)
19 Java8概述(Java8概述+lambda表达式+函数式接口+方法引用+Stream+新时间API)
55 8