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++;
}
}
上面的迭代方法就是外部迭代。
外部迭代缺点:
很难抽象出复杂操作
本质上讲是串行化操作。
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. 操作整合
collect(toList())方法由Stream里的值生成一个列表
map可以将一种类型的值转换成另一种类型。
遍历并检查其中的元素时,可用filter
如果有一个包含了多个集合的对象希望得到所有数字的集合,我们可以用flatMap
max和min
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());
原始代码和流链式调用相比,有以下缺点:
代码可读性差,隐匿了真正的业务逻辑
需要设置无关变量来保存中间结果
效率低,每一步都及早求值生成新集合
难于并行化处理
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举例子:
Java-函数式编程(三)流(Stream)的更多相关文章
- Java函数式编程:三、流与函数式编程
本文是Java函数式编程的最后一篇,承接上文: Java函数式编程:一.函数式接口,lambda表达式和方法引用 Java函数式编程:二.高阶函数,闭包,函数组合以及柯里化 前面都是概念和铺垫,主要讲 ...
- Java函数式编程原理以及应用
一. 函数式编程 Java8所有的新特性基本基于函数式编程的思想,函数式编程的带来,给Java注入了新鲜的活力. 下面来近距离观察一下函数式编程的几个特点: 函数可以作为变量.参数.返回值和数据类型. ...
- 《深入理解Java函数式编程》系列文章
Introduction 本系列文将帮助你理解Java函数式编程的用法.原理. 本文受启发于JavaOne 2016关于Lambda表达式的相关主题演讲Lambdas and Functional P ...
- Java函数式编程:二、高阶函数,闭包,函数组合以及柯里化
承接上文:Java函数式编程:一.函数式接口,lambda表达式和方法引用 这次来聊聊函数式编程中其他的几个比较重要的概念和技术,从而使得我们能更深刻的掌握Java中的函数式编程. 本篇博客主要聊聊以 ...
- paip.提升效率---filter map reduce 的java 函数式编程实现
#paip.提升效率---filter map reduce 的java 函数式编程实现 ======================================================= ...
- Java函数式编程:一、函数式接口,lambda表达式和方法引用
Java函数式编程 什么是函数式编程 通过整合现有代码来产生新的功能,而不是从零开始编写所有内容,由此我们会得到更加可靠的代码,并获得更高的效率 我们可以这样理解:面向对象编程抽象数据,函数式编程抽象 ...
- Java 函数式编程(Lambda表达式)与Stream API
1 函数式编程 函数式编程(Functional Programming)是编程范式的一种.最常见的编程范式是命令式编程(Impera Programming),比如面向过程.面向对象编程都属于命令式 ...
- 菜鸡的Java笔记 第三十六 - java 函数式编程
StudyLambda Lambda 指的是函数式编程,现在最为流行的编程模式为面向对象,很多的开发者并不认可面向对象,所以很多的开发者宁愿继续使用 C 语言进行开发,也不愿意使用java,c+ ...
- Java 函数式编程
由 JS 转 Java,写惯了 React,习惯了函数式,因此转 Java 时也是先学函数式. 语法糖「Syntactic Sugar」 起初,Java 的函数式看起来是匿名类的一个语法糖. Stre ...
- 精练代码:一次Java函数式编程的重构之旅
摘要:通过一次并发处理数据集的Java代码重构之旅,展示函数式编程如何使得代码更加精练. 难度:中级 基础知识 在开始之前,了解"高阶函数"和"泛型"这两个概念 ...
随机推荐
- 企业项目开发--cookie(3)
此文已由作者赵计刚授权网易云社区发布. 欢迎访问网易云社区,了解更多网易技术产品运营经验. 2.2.3.AdminController 1 package com.xxx.web.admin; ...
- 自己动手python打造渗透工具集
难易程度:★★★阅读点:python;web安全;文章作者:xiaoye文章来源:i春秋关键字:网络渗透技术 前言python是门简单易学的语言,强大的第三方库让我们在编程中事半功倍,今天我们就来谈谈 ...
- Apache VirtualHost的配置
自从电脑更换为mac后, 一直没有时间去配置php的环境.导致每次要更改php代码的时候, 都是本地更改,然后直接推送到服务器上运行 这样的开发和测试及其耗时且繁琐, 所以早上特地决定弄好mac下的p ...
- Storm入门示例
开发Storm的第一步就是设计Topology,为了方便开发者入门,首先我们设计一个简答的例子,该例子的主要的功能就是把每个单词的后面加上Hello,World后缀,然后再打印输出,整个例子的Topo ...
- jQuery的ajax的post请求json格式无法上传空数组
问题描述:在和后端对接时,使用jquery的ajax的post方式向后端传递一序列约定好格式的对象数组.遇到了一个现象:如果对象中的数组是空数组,那么在请求参数中是不会出现的. 以下是数据的对比: ...
- iOS开发笔记-图标和图片大小官方最新标准
这两天开发iOS app用到了Tab bar,然后随便切了点图标放上去发现效果极差.于是乎,开始查找苹果官方给的标准.搜索一番后,看到了一篇博文,但其内容与iOS人机交互指南最新版内容不符. 故此,在 ...
- Jmeter测试Mysql数据库-入门篇
一.jmter配置数据库 1.在配置jmter之前需要先安装数据库连接池驱动,进入到官方下载页面https://dev.mysql.com/downloads/connector/j/,下载对应的驱动 ...
- JAVA面试精选【Java算法与编程二】
在面试中,算法题目是必须的,通过算法能够看出一个程序员的编程思维,考察对复杂问题的设计与分析能力,对问题的严谨性都能够体现出来.算法是一系列解决问题的清晰指令,也就是说,能够对一定规范的输入,在有限时 ...
- ASP.NET Core 中使用 GrayLog 记录日志
使用 UDP 协议发送日志 自定义好的查询 key 存储数据,尽量不要使用 graylog2-server 服务端格式化日志再存储 Ubuntu 安装服务端 sudo apt-get update & ...
- Maven 的基本配置与使用
什么是Maven Maven是基于项目对象模型(POM),可以通过一小段描述信息来管理项目的构建,报告和文档的软件项目管理工具. 发文时,绝大多数开发人员都把 Ant 当作 Java 编程项目的标准构 ...