转 java 8 lamba stream
一直在写中间件相关的代码,提供SDK给业务方使用,但很多业务方还一直停留在1.7版本,迟迟不升级,为了兼容性,不敢在代码中使用Java8的一些新特性,比如Stream之类的,虽然不能用,但还是要学一下。
Stream 是什么
Stream 是Java 8中添加的一个新特性,它与 java.io 包里的 InputStream 和 OutputStream 是完全不同的概念。它借助于 Lambda 表达式,可以让你以一种声明的方式处理数据,可以极大提高Java程序员的生产力,让程序员写出高效率、干净、简洁的代码。
Stream Demo
直接上Demo,感受一下
List<String> myList = Arrays.asList("a", "b", "c", "d", "e");
myList.stream()
.filter(s -> s.startsWith("1"))
.map(String::toUpperCase)
.sorted()
.forEach(System.out::println);
Stream 如何工作
当使用一个流的时候,通常包括三个基本步骤:
获取一个数据源(source)
数据转换
执行操作获取想要的结果
每次转换原有 Stream 对象不改变,返回一个新的 Stream 对象(可以有多次转换),这就允许对其操作可以像链条一样排列,变成一个管道,如下图所示。
在Stream中,分为两种操作
中间操作
结束操作
中间操作返回Stream,终端操作返回void或者非Stream结果,在demo中, filter
、 map
、 sorted
都算是中间操作,而 forEach
是一个结束操作。
Stream 如何生成
创建Stream的方式很多,最常见的是从Collections,List 和 Set中生成
List<String> myList = Arrays.asList("a1", "a2", "b1", "c2", "c1");
Stream<String> stream = myList.stream()
在对象myList上调用方法 stream() 返回一个常规对象Stream。
也可以从一堆已知对象中生成。
Stream<String> stream = Stream.of("a1", "a2", "a3")
当然了,还有其它方式:
Collection.stream()
Collection.parallelStream()
BufferedReader.lines()
Files.walk()
BitSet.stream()
Random.ints()
JarFile.stream()
....
常规操作
forEach
forEach
方法接收一个 Lambda 表达式,用来迭代流中的每个数据
Stream.of(1, 2, 3).forEach(System.out::println);
// 1
// 2
// 3
map
map
用于映射每个元素到对应的结果
Stream.of(1, 2, 3).map( i -> i*i).forEach(System.out::println);
// 1
// 4
// 9
filter
filter
用于通过设置的条件过滤出元素
Stream.of(1, 2, 3).filter( i -> i == 1).forEach(System.out::println);
// 1
limit
limit
用于用于获取指定数量的流
Stream.of(1, 2, 3, 4, 5).limit(2).forEach(System.out::println);
// 1
// 2
sorted
sorted
用于对流进行排序
Stream.of(4, 1, 5).sorted().forEach(System.out::println);
// 1
// 4
// 5
Match
有三个 match 方法,从语义上说:
allMatch:Stream 中全部元素符合传入的 predicate,返回 true
anyMatch:Stream 中只要有一个元素符合传入的 predicate,返回 true
noneMatch:Stream 中没有一个元素符合传入的 predicate,返回 true
它们都不是要遍历全部元素才能返回结果。例如 allMatch 只要一个元素不满足条件,就 skip 剩下的所有元素,返回 false。
boolean result = Stream.of("a1", "a2", "a3").allMatch(i -> i.startsWith("a"));
System.out.println(result);
// true
reduce
reduce
方法根据指定的函数将元素序列累积到某个值。此方法有两个参数:
起始值
累加器函数。
如果有一个List,希望得到所有这些元素和一些初始值的总和。
int result = Stream.of(1, 2, 3).reduce(20, (a,b) -> a + b);
System.out.println(result);
// 26
collect
Collectors类中提供了功能丰富的工具方法
toList
toSet
toCollection
toMap
...
而这些方法,都需要通过 collect
方法传入。
Set<Integer> result = Stream.of(1, 1, 2, 3).collect(Collectors.toSet());
System.out.println(result);
// [1, 2, 3]
collect
可以把Stream数据流转化为Collection对象,
骚技巧
for循环
除了常规的对象Stream,还有一些有特殊类型的Stream,用于处理基本数据类型int、long和double,它是IntStream、LongStream和DoubleStream。
比如可以使用IntStream.range()来代替常规的for循环。
IntStream.range(1, 4).forEach(System.out::println);
随机数
Random的ints方法可以返回一个随机数据流,比如返回1到100的10个随机数。
Random random = new Random();
random.ints(1, 100).limit(10).forEach(System.out::println);
大小写转化
List<String> output = wordList.stream()
.map(String::toUpperCase)
.collect(Collectors.toList());
Stream 特点
总之,Stream 的特性可以归纳为:
无存储
Stream并不是一种数据结构,它只是某种数据源的一个视图
安全性
对Stream的任何修改都不会修改背后的数据源,比如对stream执行过滤操作并不会删除被过滤的元素,而是会产生一个不包含被过滤元素的新Stream。
惰式执行
Stream上的操作并不会立即执行,只有等到用户真正需要结果的时候才会执行。
一次性
Stream只能被“消费”一次,一旦遍历过就会失效,就像容器的迭代器那样,想要再次遍历必须重新生成。
lambda
所有 Stream 的操作必须以 lambda 表达式为参数
转 java 8 lamba stream的更多相关文章
- 详解Java 8中Stream类型的“懒”加载
在进入正题之前,我们需要先引入Java 8中Stream类型的两个很重要的操作: 中间和终结操作(Intermediate and Terminal Operation) Stream类型有两种类型的 ...
- java.io.IOException: Stream closed
今天在做SSH项目的时候,出现了这个错误.百思不得其解,网上的答案都不能解决我的问题-.. 后来,一气之下就重新写,写了之后发现在JSP遍历集合的时候出错了. <s:iterator value ...
- Java基础(十一) Stream I/O and Files
Java基础(十一) Stream I/O and Files 1. 流的概念 程序的主要任务是操纵数据.在Java中,把一组有序的数据序列称为流. 依据操作的方向,能够把流分为输入流和输出流两种.程 ...
- java.io.IOException: Stream closed解决办法
1.出现这个bug的大体逻辑代码如下: private static void findMovieId() throws Exception { File resultFile = new File( ...
- Java 8创建Stream流的5种方法
不知不觉间,Java已经发展到13了,来不及感慨时间过得真的太快了,来不及学习日新月异的技术更新,目前大多数公司还是使用的JDK8版本,一方面是版本的稳定,另一方面是熟悉,所以很多公司都觉得不升级也挺 ...
- java中的Stream流
java中的Stream流 说到Stream便容易想到I/O Stream,而实际上,谁规定"流"就一定是"IO流"呢?在Java 8中,得益于Lambda所带 ...
- java新特性stream
java新特性stream,也称为流式编程. 在学习stream之前先了解一下java内置的四大函数 第一种函数式函数,后面是lambda表达式写法 /*Function<String,Inte ...
- Java基础系列-Stream
原创作品,可以转载,但是请标注出处地址:https://www.cnblogs.com/V1haoge/p/10748817.html 一.概述 Stream操作简称流操作,这里的流与IO流毫无关系, ...
- Java 8中Stream API学习笔记
1)函数式编程的优势和劣势分别是什么?优势:①不可变性 ②并行操作 ③执行顺序更灵活 ④代码更加简洁纯粹的函数式编程,变量具有不可变性,同一个参数不会在不同场景下得出不同的结果,因此大大增强了系统的稳 ...
随机推荐
- 限制textarea 文本框的长度(收集了几个办法)
文章参考地址:http://blog.csdn.net/nhconch/article/details/4223076 需求背景: 前几天,因为客户有一个需求说,需要对一个评论(一个textarea文 ...
- new.target元属性 | 分别用es5、es6 判断一个函数是否使用new操作符
函数内部有两个方法 [[call]] 和 [[construct]] (箭头函数没有这个方法),当使用new 操作符时, 函数内部调用 [[construct]], 创建一个新实例,this指向这个实 ...
- python面试题&练习题之函数
1.写函数,接收两个数字参数,返回最大值例如:传入:10,20返回:20 def res_max(number1,number2): l1 = [] l1.append(number1) l1.app ...
- Supercharging your ETL with Airflow and Singer
转自:https://www.stitchdata.com/blog/supercharging-etl-with-airflow-and-singer/ singer 团队关于singer 与air ...
- AcWing 95 费解的开关
目录 前言 题目链接 思路 代码 前言 博客咕咕咕了好久了,是时候写一下了 题目链接 AcWing 95 费解的开关 思路 首先可以看出 1.每一个位置顶多只会操作一次.因为如果操作两次的话,相当于不 ...
- 在 Mac、Linux、Windows 下Go交叉编译
Golang 支持交叉编译,在一个平台上生成另一个平台的可执行程序,最近使用了一下,非常好用,这里备忘一下. Mac 下编译 Linux 和 Windows 64位可执行程序 CGO_ENABLED= ...
- 【技术博客】MySQL和Django常用操作
MySQL和Django是搭建网站常用的配置之一,在此记录一下在Windows系统搭建网站时MySQL以及Django常用的操作. MySQL MySQL的SQL语句不区分大小写,推荐将保留字大写,数 ...
- RPGMaker MV 入门教程
RPG Maker是一个十分优秀的rpg游戏制作引擎 恩 一个被定义为游戏的游戏引擎 可以用来十分便捷的制作rpg游戏 有兴趣的可以尝试一下 满足自己想做游戏的愿望. Step1 决定你的RPG形 ...
- SpringCloud之application.properties和bootstrap.properties区别
Spring是有上下文一说的,也叫Application Context,Application Context又是有父子关系的,所以必须要理解ApplicationContext是什么.Spring ...
- Spring Cloud-Eureka 服务注册中心
Eureka 是 Netflix 开发的,一个基于 REST 服务的,服务注册与发现的组件 它主要包括两个组件:Eureka Server 和 Eureka Client Eureka Client: ...