Java ParallelStream
ParallelStream 处理数据
Stream 接口提供了parallelStream方法来将集合转换为并行流。即将一个集合分为多个数据块,并用不同的线程分别处理每个数据块的流。
并且使用parallelStream 时无需担心内部变量控制,线程数量等问题。
如使用并行流计算1至100000累加之和:
- 最后一次parallel或sequential调用会影响整个流水线,即如下例子中会并行执行。
- parallelStream使用得默认核心数为Runtime.getRuntime().availableProcessors() - 1。
可通过配置java.util.concurrent.ForkJoinPool.common.parallelism改变默认的核心数。
Stream.iterate(1L, param1 -> Math.addExact(param1, 1))
.limit(100000)
.parallel()
.sequential()
.parallel()
.reduce(0L, Math::addExact)
.longValue();
parallelStream 性能分析
通常我们认为在数据量到达一定程度时,使用多线程计算会获得更好的性能。但实际效果应该在测量比较之后才直到。
使用并行流和顺序流别计算1至100000 的累加之和,在我的四核英特尔机器上运行结果如下:
long start = System.currentTimeMillis();
Stream.iterate(1L, param1 -> Math.addExact(param1, 1))
.limit(100000)
.parallel()
.reduce(0L, Math::addExact)
.longValue();
System.out.println(String.format("Parallel accumulate sum, used %d ms.", System.currentTimeMillis() - start));
start = System.currentTimeMillis();
LongStream.rangeClosed(1, 100000)
.reduce(0L, Math::addExact);
System.out.println(String.format("Sequential accumulate sum, used %d ms.", System.currentTimeMillis() - start));
Parallel accumulate sum, used 64 ms.
Sequential accumulate sum, used 8 ms.
通过以上结果可以看到,并行流计算的耗时竟然是顺序流的好几倍,这与我们的预期结果差距十分的大。
要想明白这差距的原因,首先得明白影响上面并行流的速度的因素有那些:
- 元素是否容易拆分为多个数据块, 很明显Iterate 很难拆分为多个独立数据块,因为每次应用这个函数都要依赖于前一个元素。
- 元素是否频繁拆装箱, 流中Long -> long 频繁拆装箱也影响了效率。而LongStream 中并没有这个消耗。
修复上面两个影响并行流的速度的问题后,重新运行结果如下:
long start = System.currentTimeMillis();
LongStream.rangeClosed(1, 100000)
.parallel()
.reduce(0L, Math::addExact);
System.out.println(String.format("Parallel accumulate sum, used %d ms.", System.currentTimeMillis() - start));
start = System.currentTimeMillis();
LongStream.rangeClosed(1, 100000)
.reduce(0L, Math::addExact);
System.out.println(String.format("Sequential accumulate sum, used %d ms.", System.currentTimeMillis() - start));
Parallel accumulate sum, used 7 ms.
Sequential accumulate sum, used 3 ms.
并行流的速度得到了很大提升,这表明并行化时需要使用正确的数据结构。
但是顺序流的速度却仍然更快,这说明并行化也是有代价的,如下:
- 内核之间交换数据的花销较大。
- 要保证在内核中的处理时间大于内核间的数据交换时间,即数据到达一定的量级。
而并行过程需要对流要递归划分,再把每个子流的归纳操作分配到不同的线程,最后把这些操作的结果合并成一个值。
在子流归纳操作时间过短时,并行化并没有带来性能提升,反而是更加慢了。
再将数据提升至上亿级别进行运算,并行流终于取得了一些领先。
long start = System.currentTimeMillis();
LongStream.rangeClosed(1, 100000000)
.parallel()
.reduce(0L, Math::addExact);
System.out.println(String.format("Parallel accumulate sum, used %d ms.", System.currentTimeMillis() - start));
start = System.currentTimeMillis();
LongStream.rangeClosed(1, 100000000)
.reduce(0L, Math::addExact);
System.out.println(String.format("Sequential accumulate sum, used %d ms.", System.currentTimeMillis() - start));
Parallel accumulate sum, used 79 ms.
Sequential accumulate sum, used 264 ms.
高效使用ParallelStream
关于在什么地方使用parallelStream 没有绝对的建议,而是只能做定性分析。下列是一些可能影响性能的地方:
- 测量比较,并行流并不都比顺序流快。
- 避免拆装箱,这对性能有较大影响。可使用原始类型IntStream, LongStream等。
- 依赖元素顺序的操作,并行性能较差。如findAny()性能会优于findFirst(),因为它不依赖于顺序。
- 数据量大小,估算一个元素通过流水线的大概处理时间,得到处理完整个集合的处理时间。
- 流是否易于拆分,如ArrayList 比LinkedList 更易于拆分,前者无需遍历,后者需要遍历之后才能拆分。
- 终端操作时,合并操作的代价大小(例如Collector中的combiner方法)。
Fork/Join
ParallelStream流背后使用的基础架构是Java 7中引入的Fork/Join分支合并框架。
分支/合并框架的目的是以递归方式将可以并行的任务拆分成更小的任务,然后将每个子任务的结果合并起来生成整体结果。
这其实就是分治算法的并行版本。
Java ParallelStream的更多相关文章
- 深入浅出parallelStream
援引:http://blog.csdn.net/u011001723/article/details/52794455 感谢作者的分享!感谢作者为JDK8的学习所做的努力. about Stream ...
- Spark案例分析
一.需求:计算网页访问量前三名 import org.apache.spark.rdd.RDD import org.apache.spark.{SparkConf, SparkContext} /* ...
- 【Java】关于Java8 parallelStream并发安全的思考
背景 Java8的stream接口极大地减少了for循环写法的复杂性,stream提供了map/reduce/collect等一系列聚合接口,还支持并发操作:parallelStream. 在爬虫开发 ...
- java并行之parallelStream与CompletableFuture比较
1. import java.util.Arrays; import java.util.List; import java.util.concurrent.CompletableFuture; im ...
- Java 8里 Stream和parallelStream的区别
Java中Stream和parallelStream,前者是单管,后者是多管,运行时间上做一个小对比,直接上代码: /** * * @author zhangy6 * <p>对比Strea ...
- Java Arrays.asList(0,1,2,3,4,5,6,7,8,9).parallelStream().forEach 进行循环获取HttpServletRequest的为Null的解决方案
Arrays.asList(0,1,2,3,4,5,6,7,8,9).parallelStream().forEach() parallelStream是并行执行流的每个元素,也就是多线程执行,这样就 ...
- JAVA使用并行流(ParallelStream)时要注意的一些问题
https://blog.csdn.net/xuxiaoyinliu/article/details/73040808
- Java 8函数编程轻松入门(五)并行化(parallel)
1.并发与并行的区别 并发: 一个时间段内有几个程序都处于已启动到运行完毕之间,且这几个程序都是在同一个处理机上运行.但在任一个时刻点只有一个程序在处理机上运行 并行: 在同一个时刻,多核处理多个任务 ...
- Java 8 指南
Benjamin Winterberg “Java is still not dead—and people are starting to figure that out.” 欢迎阅读我对 Java ...
随机推荐
- 1、mysql基础入门(2)
1.4.常用非关系型数据库产品介绍: 1.Memcached(key-value)数据库:
- layui 列合并相同内容
table.render({ elem: '#tbdata', method: 'post', data: jsonData, height: temphei, limit: 20, limits: ...
- POJ 1222 高斯消元更稳
大致题意: 有5*6个灯,每个灯只有亮和灭两种状态,分别用1和0表示.按下一盏灯的按钮,这盏灯包括它周围的四盏灯都会改变状态,0变成1,1变成0.现在给出5*6的矩阵代表当前状态,求一个能全部使灯灭的 ...
- 关于windows11的0x800f0950语言包安装失败
最近windows11的风头很热,作为爱折腾的人,当然要去搞一搞啦.搞好了以后我发现中文语言的拓展包是无法安装的,于是我找到了3个办法,当然如果想100%成功的话我建议直接跳到第三个,如果你不嫌累,指 ...
- SpringBoot拦截器及源码分析
1.拦截器是什么 java里的拦截器(Interceptor)是动态拦截Action调用的对象,它提供了一种机制可以使开发者在一个Action执行的前后执行一段代码,也可以在一个Action执行前阻止 ...
- 案例分享:Qt西门子机床人机界面以及数据看板定制(西门子通讯,mysql数据库,生产信息,参数信息,信息化看板,权限控制,播放器,二维图表,参数调试界面)
若该文为原创文章,转载请注明原文出处本文章博客地址:https://blog.csdn.net/qq21497936/article/details/118685521 长期持续带来更多项目与技术分享 ...
- 资源:CentOS下载地址资源
新版本系统镜像下载(当前最新是CentOS 7.4版本) CentOS官网 官网地址 http://isoredirect.centos.org/centos/7.4.1708/isos/x86_64 ...
- java基础---设计模式(1)
出处:https://blog.csdn.net/zhangerqing/article/details/8194653 java的设计模式分为三大类 创建型模式: 工厂方法模式.抽象工厂模式.单例模 ...
- Nacos入门学习&实践
文中涉及到了一些模块代码没有给出,我一并上传到github了,可以整个项目clone下来进行调试. 地址:https://github.com/stronglxp/springcloud-test 1 ...
- 鸟哥的Linux私房菜基础学习篇--进程(process)一 归纳总结
权限!权限!权限! 没有权限,一些资源你是没办法使用的.在Linux中cat filename,结果屏幕显示了filename的内容,为什么你能看见,而我不能?权限.与UID/GID有关,与文件的属性 ...