JAVA8新特性–集合流操作Stream

Stream类全路径为:java.util.stream.Stream
对Stream的描述,引用其他文章中觉得比较好的介绍:

Java 8 中的 Stream 是对集合(Collection)对象功能的增强,它专注于对集合对象进行各种非常便利、高效的聚合操作(aggregate operation),或者大批量数据操作 (bulk data operation)。Stream API 借助于同样新出现的 Lambda 表达式,极大的提高编程效率和程序可读性。
即Stream的原理:

这种风格将要处理的元素集合看作一种流, 流在管道中传输, 并且可以在管道的节点上进行处理, 比如筛选, 排序,聚合等。元素流在管道中经过中间操作(intermediate operation)的处理,最后由最终操作(terminal operation)得到前面处理的结果。
集合有两种方式生成流:
– stream() − 为集合创建串行流。
– parallelStream() − 为集合创建并行流。

并发模式(parallelStream)能够充分利用多核处理器的优势,使用 fork/join 并行方式来拆分任务和加速处理过程。java中也提供了多种并行处理的方式,看到一片文章对各几种并行的方式性能进行了测试,可以参考下:java中几种并行方式的性能分析。

Stream的的中间操作(intermediate)和最终操作(terminal)都包含哪些方法可以从类结构中看到:

上面截图基本包含了Strram的所有方法。

中间操作(intermediate)主要有以下方法(此类型的方法返回的都是Stream对象):
map (mapToInt, flatMap 等)、 filter、 distinct、 sorted、 peek、 limit、 skip、 parallel、 sequential、 unordered

终端操作(terminal)主要有以下方法:
forEach、 forEachOrdered、 toArray、 reduce、 collect、 min、 max、 count、 anyMatch、 allMatch、 noneMatch、 findFirst、 findAny、 iterator

此处按照类结构(截图)中的方法的顺序(以下例子中使用了Lambda表达式及方法引用,不了解的请戳:Lambda介绍,方法引用),进行介绍:

filter:通过设置条件来过滤元素。
List

list.stream()
.filter((s)->s.contains(“a”))
.forEach(s -> System.out.println(s));
1
2
3
4
5
以上代码使用filter方法过滤出只包含”a”的元素,然后通过forEach将满足条件的元素遍历出来。输出如下:

aaa
a2a
a3a
map:就是将对应的元素使用给定方法进行转换。
list.stream()
.filter((s)->s.contains(“a”))
.map((s)-> s + “—map”)
.forEach(s -> System.out.println(s));
1
2
3
4
5
在filter的基础上,给每个元素后面添加字符串”—map”,输出如下:

aaa—map
a2a—map
a3a—map
mapToInt:和map方法进行一样的操作,但是转换函数必须返回int类型
mapToLong:和map方法进行一样的操作,但是转换函数必须返回long类型
mapToDouble:和map方法进行一样的操作,但是转换函数必须返回double类型
此处以mapToInt进行演示:
list.stream()
.filter((s)->s.contains(“a”))
.mapToInt((s)-> s.hashCode())
.forEach(s -> System.out.println(s));
1
2
3
4
在filter的基础上,将每个元素转换为其hashCode。输出为:

96321
94864
94895
flatMap:如果流的元素为数组或者Collection,flatMap就是将每个Object[]元素或Collection

List

map==java.util.stream.ReferencePipeline&Head@50040f0c
map==java.util.stream.ReferencePipeline&Head@2dda6444
map==java.util.stream.ReferencePipeline&Head@5e9f23b4
flatMap==aa
flatMap==bb
flatMap==cc
flatMap==dd
flatMap==ee
flatMap==ff
可以看出map就是将数组流直接返回,flatMap是将数组流中的每个元素都返回。flatMapToInt,flatMapToLong,flatMapToDouble类似,只不过返回的是对应的类型的流,此处不做演示。

distinct:将集合中的元素去重。
List

aaa
ddd
bbb
sorted:将集合中的元素排序。
List

1
2
3
4
可以按照自定义排序:

integerList.stream()
.sorted((s1,s2)->s2.compareTo(s1))
.forEach(s-> System.out.println(s));
1
2
3
输出如下:

4
3
2
1
peek:生成一个包含原Stream的所有元素的新Stream,同时会提供一个消费函数即引用的方法A,当Stream每个元素被消费的时候都会先执行新Stream给定的方法A。peek是中间操作,如果pee后没有最终操作,则peek不会执行。
integerList.stream()
.peek(s-> System.out.println(“peek = “+s));
1
2
此时没有输出,代码改为:

integerList.stream()
.peek(s-> System.out.println(“peek = “+s))
.forEach(s-> System.out.println(“forEach = “+s));
1
2
3
输出如下:

peek = 1
forEach = 1
peek = 2
forEach = 2
peek = 3
forEach = 3
peek = 4
forEach = 4
limit:返回Stream的前n个元素。
integerList.stream()
.limit(1)
.forEach(s-> System.out.println(s));
1
2
3
输出为:

1
skip:删除Stream的前n个元素。
integerList.stream()
.skip(1)
.forEach(s-> System.out.println(s));
1
2
3
输出如下:

2
3
4
forEach:遍历Stream中的每个元素,前面每个例子都有使用,此处不再演示。
forEachOrdered:遍历Stream中的每个元素。
区别:
在串行流(stream)中没有区别,在并行流(parallelStream)中如果数据源是有序集合,forEachOrdered输出顺序与数据源中顺序一致,forEach则是乱序。
看下使用forEach:

integerList.stream()
.forEach(s-> System.out.println(s));
1
2
输出(多次测试,每次结果都不一样):

3
1
4
2
再看使用forEachOrdered:

integerList.parallelStream()
.forEachOrdered(s-> System.out.println(s));
1
2
输出(测试多次,每次都是这个结果,与integerList中的元素顺序一致):

2
1
3
4
toArray:将流转换为Object[]或者指定类型的数组。
Object[] array = integerList.stream().toArray();
String[] strArr = integerList.stream().toArray(String[]::new);
1
2
reduce:将集合中的每个元素聚合成一条数据。有三种情况:
reduce(BinaryOperator accumulator):此处需要一个参数,返回Optional对象:
Optional

15
此例中使用起始值为5,对集合中每个元素求和,可以理解为:5+2+1+3+4=15。

reduce(U identity, BiFunction

[2, 3, 4]
min:获取集合中最小值。
Integer min = integerList.stream()
.filter(a -> a > 1)
.min((Integer a, Integer b) -> a.compareTo(b))
.get();
System.out.println(min);
1
2
3
4
5
输出为:

2
max:获取集合中最大值。
Integer max = integerList.stream()
.filter(a -> a > 1)
.max((Integer a, Integer b) -> a.compareTo(b))
.get();
System.out.println(max);
1
2
3
4
5
输出为:

4
count:获取集合中元素个数
long count = integerList.stream()
.filter(a -> a > 1)
.count();
System.out.println(count);
1
2
3
4
输出为:

3
anyMatch: Stream 中任意一个元素符合传入的 predicate,返回 true
allMatch:Stream 中全部元素符合传入的 predicate,返回 true
noneMatch:Stream 中没有一个元素符合传入的 predicate,返回 true
boolean b = integerList.stream()
.anyMatch(s -> s > 0);
boolean b1 = integerList.stream()
.allMatch(s -> s > 0);
boolean b2 = integerList.stream()
.noneMatch(s -> s > 0);
System.out.println(“anyMatch = ” + b);
System.out.println(“allMatch = ” + b1);
System.out.println(“noneMatch = ” + b2);
1
2
3
4
5
6
7
8
9
输出:

anyMatch = true
allMatch = true
noneMatch = false
findFirst:如果数据源是有序集合,返回Stream 中第一个元素的Optional对象,如果是无序集合,则返回Stream 中任意一个元素的Optional对象。
Integer first = integerList.stream()
.findFirst()
.get();
System.out.println(first);
1
2
3
4
输出为:

2
findAny:返回Stream 中任意一个元素的Optional对象。
Integer any = integerList.stream()
.findAny()
.get();
System.out.println(any);
1
2
3
4
输出为:

2
Stream还有几个静态方法,返回都是Stream对象。
静态方法如下:builder(返回Builder对象)、empty、of、iterate、generate、concat。
builder:返回一个Builder对象,Builder对象在调用build()返回Stream对象。
empty:返回一个空的有序的Stream对象。
of:返回包含单个元素的有序的Stream对象。
iterate:返回一个无限元素的有序的Stream对象。需要两个参数,第一个参数为初始值,第二个参数为要引用的方法,然后会通过递归循环调用引用的方法。

Stream.iterate(2,s->s+s)
.limit(10)
.forEach(s-> System.out.println(s));
1
2
3
输出为:

2
4
8
16
32
64
128
256
512
1024
generate:返回一个无限元素的无序的的Stream对象。需要一个参数,参数为引用的方法,然后会通过循环调用引用的方法来生成元素,常用于生成常量Stream和随机元素Stream。
concat:将两个Stream连接成一个Stream。需要两个Stream作为参数,如果两个Stream都是有序的并且无论参数Stream是否是并行Stream,得到的都是有序的Stream。输出元素顺序为先输出第一个Stream的元素,然后输出第二个Stream的元素。当结果Stream关闭时候,两个参数Stream同时关闭。

Stream.concat(integerList.stream(),disList.stream())
.forEach(s-> System.out.println(s));
1
2
输出为:

2
1
3
4
11
aaa
ddd
33
bbb
ddd
aaa
————————————————
版权声明:本文为CSDN博主「昵称2019」的原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接及本声明。
原文链接:https://blog.csdn.net/bluuusea/article/details/79967039

Original: https://www.cnblogs.com/zhuawang/p/11580750.html
Author: 上校
Title: JAVA8新特性–集合流操作Stream

原创文章受到原创版权保护。转载请注明出处:https://www.johngo689.com/537793/

转载文章受原作者版权保护。转载请注明原作者出处!

(0)

大家都在看

  • 重要的不是写代码,而是读代码

    读代码是一种有益的精进方式,类似与阅读和写作,从范文得以借鉴 读代码的过程中,值得重点关注的部分 被反复使用的代码 穿越时间的代码 – 如果一段代码被使用了很久都没有被…

    Java 2023年6月15日
    077
  • Java学习-第一部分-第二阶段-第六节:泛型

    泛型 笔记目录:(https://www.cnblogs.com/wenjie2000/p/16378441.html) 泛型的理解和好处 看一个需求 请编写程序,在ArrayLi…

    Java 2023年6月16日
    072
  • Docker操作(三) 自定义镜像

    Docker操作(三)自定义镜像 自定义镜像是Docker解决配置环境问题的一个方案: Docker将整个项目连同相关的运行环境一并打包,制作成一个镜像。 当需要快速部署项目的时候…

    Java 2023年6月8日
    068
  • Spring Boot 通过RestTemplat返回pdf转图片

    Springboot 通过 RestTemplat 返回 pdf转图片 开发环境: java 8 spring boot 2.x 前后端分离 需求:前端发送pdf文件路径来,后端获…

    Java 2023年6月5日
    061
  • 一步步剖析spring bean生命周期

    关于spring bean的生命周期,是深入学习spring的基础,也是难点,本篇文章将采用代码+图文结论的方式来阐述spring bean的生命周期, 本篇文章将阐述清楚下图。 …

    Java 2023年5月30日
    0102
  • 搜索技术架构-押韵精灵

    欢迎访问我的押韵精灵,是一个在线查询押韵的诗、词、歌、字的工具网站,包含双押多押等。 上一篇谈到了如何用阿里云搭建这样的网站,本文记录下该网站的整体架构,话不多说直接上图: 搜索技…

    Java 2023年6月5日
    075
  • 戏说领域驱动设计(廿三)——工厂

    在讲解实体的章节中我们曾经介绍说过如何有效的创建实体,主要包括两种方式:工厂和构造函数。本章我们工厂进行一下详解,这种东西能有效的简化实体使用的难度,毕竟你无法通过Spring这种…

    Java 2023年6月7日
    088
  • 【力扣】523. 连续的子数组和

    给你一个整数数组 nums 和一个整数 k ,编写一个函数来判断该数组是否含有同时满足下述条件的连续子数组:子数组大小 至少为 2 ,且子数组元素总和为 k 的倍数。如果存在,返回…

    Java 2023年6月8日
    044
  • PDF转换OFD(Java实用版)

    前言: 在项目中用到了,就写一下哈 OFD简介 百度百科:https://baike.baidu.com/item/OFD/56227163?fr=aladdin OFD(Open…

    Java 2023年6月8日
    092
  • HM2022ssm-mp2【MyBatisPlus标准数据层开发】

    标准CRUD使用 新增 @Test public void testSave() { User user = new User(); user.setName("黑马程序…

    Java 2023年6月5日
    082
  • 解决idea中每次创建项目都要重复配置maven,来看全网仅需几步全局配置

    你是不是idea每次创建项目都要重新配置maven呢?为啥呢老憋屈了。因为你之前的每一次的maven配置都是在项目中去配置的 ,所以只会在本项目中生效。也就是说在项目中配置的是 &…

    Java 2023年6月5日
    096
  • java中StringBuffer和StringBuilder

    StringBuffer StringBuffer是一个容器 StringBuffer实现了Serializable,说明可以串行化 在父类中 AbstractStringBuil…

    Java 2023年6月6日
    078
  • 谈谈Raft

    本文主要参考: 极客时间-etcd 实战课 GitChat-分布式锁的最佳实践之:基于 Etcd 的分布式锁 谈到分布式协调组件,我们第一个想到的应该是大名鼎鼎的Zookeeper…

    Java 2023年6月5日
    072
  • 【LEETCODE】74、第542.题 01 矩阵

    package array.medium; import java.util.ArrayDeque; import java.util.Deque; import java.uti…

    Java 2023年6月5日
    059
  • docker 安装 mysql

    1、使用 docker 命令下载 mysql docker pull mysql:5.7 5.7是版本号,可以去 https://hub.docker.com/_/mysql?ta…

    Java 2023年6月5日
    070
  • 设计模式之策略模式应用实例(Spring Boot 如何干掉 if else)

    需求 这里虚拟一个业务需求,让大家容易理解。假设有一个订单系统,里面的一个功能是根据订单的不同类型作出不同的处理。 订单实体: service接口: 传统实现 根据订单类型写一堆的…

    Java 2023年5月30日
    078
亲爱的 Coder【最近整理,可免费获取】👉 最新必读书单  | 👏 面试题下载  | 🌎 免费的AI知识星球