JDK8中,提供了并行流和串行流,使用parallel()和sequential()来处理,parallel()为并行流sequential()为串行流,两者可以相互转换,以最后一个为准

LongStream.rangeClosed(,).sequential().parallel().reduce((x,y)->x+y);

  以上代码示例就是并行流和串行流的使用,由于parallel在后,所以是以并行流运算。

  其实JDK8的并行流和串行流并不复杂,但是想要了解其历史,就要从单线程、多线程、JDK7的fork/join框架一一说起。

  首先说单线程问题,如果是单线程,肯定是没有多线程运行快的(通常情况下如此,如果仅此是1+1的操作,多线程因为线程切换等原因,反而会更慢),况且fork/join使用的是线程窃取模式进行处理的,相比多线程更有优势,可以更好的使用内存。

  但是为什么fork/join没有被大量使用呢,主要是因为fork/join写法太繁琐,下面就举例说明fork/join的写法

1、fork/join

首先创建一个ForkJoinDemo对象,对数据进行拆分(fork)运算后汇总(join)

package com.example.jdk8demo;

import java.util.concurrent.RecursiveTask;

public class ForkJoinDemo extends RecursiveTask<Long> {
private long start;
private long end;
private static final long mm = ; public ForkJoinDemo(long start,long end){
this.start = start;
this.end = end;
} @Override
protected Long compute() {
long length = end - start;
if(length <= mm){
long sum = ;
for (long i=start;i<= end;i++){
sum +=i;
}
return sum;
}else{
long mid = (start+end)/;
ForkJoinDemo left = new ForkJoinDemo(start,mid);
left.fork();
ForkJoinDemo right = new ForkJoinDemo(mid+,end);
right.fork();
return left.join() + right.join();
}
}
}

  测试方法:

public void test1(long num){
Instant start = Instant.now();
ForkJoinPool forkJoinPool = new ForkJoinPool();
ForkJoinTask<Long> forkJoinTask = new ForkJoinDemo(,num);
long sum = forkJoinPool.invoke(forkJoinTask);
Instant end = Instant.now();
log.info("fork/join运行时间【{}】,运行结果【{}】",Duration.between(start,end).toMillis(),sum);
}

  可以发现使用fork/join的写法非常麻烦

JDK8提供的串行流和并行流的操作就非常方便

2、串行流

    public void test3(long num){
Instant start = Instant.now();
OptionalLong optionalLong = LongStream.rangeClosed(,num).sequential().reduce((x,y)->x+y);
Instant end = Instant.now();
log.info("串行流运行时间【{}】,运行结果【{}】",Duration.between(start,end).toMillis(),optionalLong.getAsLong());
}

3、并行流

    public void test4(long num){
Instant start = Instant.now();
OptionalLong optionalLong = LongStream.rangeClosed(,num).parallel().reduce((x,y)->x+y);
Instant end = Instant.now();
log.info("并行流运行时间【{}】,运行结果【{}】",Duration.between(start,end).toMillis(),optionalLong.getAsLong());
}

4、为了演示执行时间,再添加一个单线程测试

    public void test2(long num){
long sum = ;
Instant start = Instant.now();
for(int i=;i<=num;i++){
sum+=i;
}
Instant end = Instant.now();
log.info("单线程for循环运行时间【{}】,运行结果【{}】",Duration.between(start,end).toMillis(),sum);
}

5、测试

  由于都是数据的累加操作,因此多线程由于线程切换等原因,会造成比单线程执行慢的假象,为了排除这一假象,直接累加到20亿的运行时间作为参考

    @Test
public void test(){
long num = **10000L;
test1(num);
test3(num);
test4(num);
test2(num);
}

运行结果:

单线程运行了一分钟,还没有出结果

fork/join运行时间3899毫秒,串行流2081毫秒,并行流1532毫秒,可见性能提升还是非常明显的。

JDK8--07:并行流与串行流的更多相关文章

  1. 【Java8新特性】关于并行流与串行流,你必须掌握这些!!

    写在前面 提到Java8,我们不得不说的就是Lambda表达式和Stream API.而在Java8中,对于并行流和串行流同样做了大量的优化.对于并行流和串行流的知识,也是在面试过程中,经常被问到的知 ...

  2. Java8的新特性--并行流与串行流

    目录 写在前面 Fork/Join框架 Fork/Join框架与传统线程池的区别 传统的线程池 Fork/Join框架 Fork/Join框架的使用 Java8中的并行流 写在前面 我们都知道,在开发 ...

  3. Java8新特性 并行流与串行流 Fork Join

    并行流就是把一个内容分成多个数据块,并用不同的线程分 别处理每个数据块的流. Java 8 中将并行进行了优化,我们可以很容易的对数据进行并 行操作. Stream API 可以声明性地通过 para ...

  4. 三、并行流与串行流 Fork/Join框架

    一.并行流概念: 并行流就是把一个内容分成多个数据块,并用不同的线程分别处理每个数据块的流. java8中将并行进行了优化,我们可以很容易的对数据进行并行操作.Stream API可以声明性的通过pa ...

  5. Java8新特性 - 并行流与串行流

    并行流就是把一个内容分成多个数据块,并用不同的线程分别处理每个数据块的流. Java8中将并行进行了优化,我们可以很容易的对数据进行并行操作.Stream API可以声明性地通过parallel()和 ...

  6. ForkJoin、并行流计算、串行流计算对比

    ForkJoin 什么是 ForkJoin ForkJoin 是一个把大任务拆分为多个小任务来分别计算的并行计算框架 ForkJoin 特点:工作窃取 这里面维护的都是双端队列,因此但其中一个线程完成 ...

  7. Silverlight并行下载与串行下载

    思路清晰后仅仅只需百来行代码便可轻松编写出一套完整的资源动态下载组件- SerialDownloader和ParallelDownloader,它们共用一个完成资源表,且串行下载集成了优先机制(Dow ...

  8. for循环与串行化、并行化Stream流性能对比

    第四章 并行化Stream流 关注公众号(CoderBuff)回复"stream"获取<Java8 Stream编码实战>PDF完整版. <Java8 Strea ...

  9. iOS:GCD理解1(同步-异步、串行-并行)

    1.并行-异步(ST1与ST2抢占资源) 1-1).获取 并行(全局)队列 ,DISPATCH_QUEUE_PRIORITY_DEFAULT 为默认优先级. dispatch_queue_t queu ...

随机推荐

  1. Rocket - util - Annotations

    https://mp.weixin.qq.com/s/7C8ZmPpwAqFqyKjL9K40Fg   介绍util中定义的注解(Annotations).   ​​   1. Annotation ...

  2. jchdl-GSL-实例 - 使用Intellij IDEA创建Mux

    https://mp.weixin.qq.com/s/yP9xKeg0iHJChuMPzxdJtA https://github.com/wjcdx/jchdl/blob/master/src/org ...

  3. 题解 P5329 【[SNOI2019]字符串】

    用栈的做法来水一发. 首先我们有一个暴力的做法,枚举每个被删除的字符,然后排序输出,时间复杂度:$ O ( N \times N \times LogN ) $ . 然后我们观察一下数据,发现有一个数 ...

  4. Vue中将网址、动态网址转为二维码

    1. 首先需要安装相关的依赖包 npm install qrcodejs2 --save 或者 npm install qrcode2 --save 这里选择第二种方式进行安装,如图: 2.templ ...

  5. Java实现 洛谷 P1023 税收与补贴问题

    import java.util.Scanner; public class Main { public static void main(String[] args){ Scanner in = n ...

  6. 支持MMDVM的DMR手台

    只要是数字机,都支持,但是,有便宜又好用的吗?当然有,店主做那么久肯定知道哪些机好用 1.MD760(UV双段) 刷固件支持#切换不同的TG,可以手动改机子的DMR ID,可以下载联系人,可以升级!关 ...

  7. 在已经编译安装好php7场景下,install gd库 with free-type (解决Call to undefined function imagettftext())

    在已经编译安装好php7场景下,install gd库 with free-type (解决Call to undefined function   imagettftext()) install g ...

  8. 滴滴HBase大版本滚动升级之旅

    桔妹导读:滴滴HBase团队日前完成了0.98版本 -> 1.4.8版本滚动升级,用户无感知.新版本为我们带来了丰富的新特性,在性能.稳定性与易用性方便也均有很大提升.我们将整个升级过程中面临的 ...

  9. vue2.0+Element UI 实现动态表单(点击按钮增删一排输入框)

    对于动态增减表单项,Element UI 官方文档表单那一节已经介绍得很清楚了,我之前没有看见,绕了很多弯路,这里针对点击按钮增删一排输入框的问题做一个总结. 效果图如下 存在一排必填的姓名与手机号, ...

  10. SuperSlide之属性targetCell介绍

    禁用页面菜单js代码: <script type="text/javascript"> document.oncontextmenu=function(){ retur ...