上篇博客我们介绍了通过CyclicBarrier使线程同步,可是上述方法存在一个问题,那就是假设一个大任务跑了2个线程去完毕。假设线程2耗时比线程1多2倍。线程1完毕后必须等待线程2完毕。等待的过程线程1没法复用。如今我们准备解决问题,我们希望线程1完毕自己的任务后能去帮助线程2完毕一部分任务。

Java7引如了Fork/Join框架能够非常好的解决问题。

Fork/Join是一个用于并行运行任务的框架,是一个把大任务切割成若干个小任务。最后汇总每一个小任务结果后得到大任务结果的框架。

fork是分叉。join是结合。

以下看张图来清晰的认识一下:

watermark/2/text/aHR0cDovL2Jsb2cuY3Nkbi5uZXQva2FuZ2xpeDFhbg==/font/5a6L5L2T/fontsize/400/fill/I0JBQkFCMA==/dissolve/70/gravity/Center" alt="">

事实上Fork/Join本质上是分治算法的一种实现。以下我们来看怎么详细使用:

ForkJoinTask:我们要使用Fork/Join框架,必须首先创建一个Fork/Join任务。它提供在任务中运行fork()和join()操作的机制,通常情况下我们不须要直接继承ForkJoinTask类。而仅仅须要继承它的子类,Fork/Join框架提供了下面两个子类:

RecursiveAction:用于没有返回结果的任务。

RecursiveTask :用于有返回结果的任务。

ForkJoinPool:ForkJoinTask须要通过ForkJoinPool来运行,任务切割出的子任务会加入到当前工作线程所维护的双端队列中,进入队列的头部。

当一个工作线程的队列里临时没有任务时,它会随机从其它工作线程的队列的尾部获取一个任务。

以下,我们相同实现考试系统抽题的样例。

GetQuestionsTask.java

public class GetQuestionsTask extends RecursiveTask<List> {

	//參数map
private Map map;
//參数list==仅仅放题型
private List questionTypeList; public GetQuestionsTask(List questionTypeList,Map map) {
this.questionTypeList = questionTypeList;
this.map=map;
} @Override
protected List compute() {
System.out.println(questionTypeList.size());
List list = new ArrayList();
if (questionTypeList.size() < 2) {
// 抽题
list = getQuestions(questionTypeList,map);
} else {
int size = questionTypeList.size();
int mid = size / 2;
GetQuestionsTask task1 = new GetQuestionsTask(
questionTypeList.subList(0, mid),map);
GetQuestionsTask task2 = new GetQuestionsTask(
questionTypeList.subList(mid, size),map);
invokeAll(task1, task2);
try {
list = groupResults(task1.get(), task2.get());
} catch (InterruptedException e) { e.printStackTrace();
} catch (ExecutionException e) { e.printStackTrace();
}
} return list;
} //合并抽题结果
private List groupResults(List list1, List list2) {
System.out.println(Thread.currentThread().getName()+"開始合并结果......");
// 合并返回结果
List list = new ArrayList();
list.addAll(list1);
list.addAll(list2);
System.out.println(Thread.currentThread().getName()+"合并结果结束......");
return list;
} // 抽题
private List getQuestions(List questTypeList,Map map) {
List list = new ArrayList();
for(int i=0;i<questTypeList.size();i++){
System.out.println(Thread.currentThread().getName()+"開始抽题......"+questionTypeList.get(i).toString());
//假数据,向list中放试题
list.add("0");
list.add("1");
list.add("2");
list.add("3");
list.add("4");
try {
Thread.sleep(1000);
} catch (InterruptedException e) { e.printStackTrace();
}
System.out.println(Thread.currentThread().getName()+"抽题结束..."+questionTypeList.get(i).toString());
}
return list;
} }

client:

//该池的线程数量不会超过0*7fff个(32767)
//池中维护着ForkJoinWorkerThread对象数组,数组大小由parallelism属性决定,parallelism默觉得处理器个数
ForkJoinPool pool = new ForkJoinPool();
GetQuestionsTask task = new GetQuestionsTask(questionTypeList, map);
pool.execute(task);
// 试题列表=task.get()
try {
List finalList = task.get();
System.out.println("终于结果个数:" + finalList.size());
} catch (InterruptedException e) { e.printStackTrace();
} catch (ExecutionException e) { e.printStackTrace();
}

总结:

Fork/Join实现了“工作窃取算法”,当一个工作线程的队列里临时没有任务时。它会随机从其它工作线程的队列的尾部获取一个任务。当然,fork/join框架的使用有一定的约束条件:

1.除了fork()  和  join()方法外,线程不得使用其它的同步工具。

线程最好也不要sleep()

2.线程不得进行I/O操作

3.线程不得抛出checked exception。

Java并发编程--Fork/Join框架使用的更多相关文章

  1. Java 并发编程 -- Fork/Join 框架

    概述 Fork/Join 框架是 Java7 提供的一个用于并行执行任务的框架,是一个把大任务分割成若干个小任务,最终汇总每个小任务结果后得到大任务结果的框架.下图是网上流传的 Fork Join 的 ...

  2. 【fork/join】java并发编程-fork/join示例

    package com.chinamobile.epic.tako.common.graphite.query.sync.impl; import com.google.common.collect. ...

  3. java 中的fork join框架

    文章目录 ForkJoinPool ForkJoinWorkerThread ForkJoinTask 在ForkJoinPool中提交Task java 中的fork join框架 fork joi ...

  4. Java 并发之 Fork/Join 框架

    什么是 Fork/Join 框架 Fork/Join 框架是一种在 JDk 7 引入的线程池,用于并行执行把一个大任务拆成多个小任务并行执行,最终汇总每个小任务结果得到大任务结果的特殊任务.通过其命名 ...

  5. JAVA中的Fork/Join框架

    看了下Java Tutorials中的fork/join章节,整理下. 什么是fork/join框架 fork/join框架是ExecutorService接口的一个实现,可以帮助开发人员充分利用多核 ...

  6. Java并发编程之——Amino框架

    Amino框架是一个采用无锁方式实现并行计算的框架,可惜的是,网上关于Amino框架的介绍甚少.根据所掌握的资料,稍微总结一下: 1. 锁机制到无锁机制 锁机制可以确保程序和数据的线程安全,但是锁是一 ...

  7. Java并发编程--4.Executor框架

    简介 Executor框架是启动,管理线程的API, 它的内部实现是线程池机制,它有很多好处,比如使任务提交和任务执行解耦合,防止this逃逸:它的主要API包括: Executor,  Execut ...

  8. Java并发编程(08):Executor线程池框架

    本文源码:GitHub·点这里 || GitEE·点这里 一.Executor框架简介 1.基础简介 Executor系统中,将线程任务提交和任务执行进行了解耦的设计,Executor有各种功能强大的 ...

  9. JAVA并发编程J.U.C学习总结

    前言 学习了一段时间J.U.C,打算做个小结,个人感觉总结还是非常重要,要不然总感觉知识点零零散散的. 有错误也欢迎指正,大家共同进步: 另外,转载请注明链接,写篇文章不容易啊,http://www. ...

随机推荐

  1. solr总结 第六部分:solr查询语法

    1.基本查询语法 q:全文查询.schema.xml里面定义了如下两块.eg q=ibm即表示org_name或者org_weisite里面出现ibm的document都可以被匹配到.KeyWords ...

  2. Thinking in UML 学习笔记(一)——建立对象模型

    一.面向对象的本质 面向对象的本质是抽象,当系统达到了超越其处理能力的程度,我们能够抽象出我们能够处理的范围来提成抽象级别,这样就能够构建更大.更复杂的系统. 现实世界和对象世界之间存在着一道沟壑,这 ...

  3. log4net概述

    log4net概貌 log4net是一个框架,用来记录日志的框架.为什么要记录日志呢?每个程序员都不能保证自己的程序完全没有错误,可是当程序已经部署的时候出现错误怎么办?我们这时候就要根据我们的日志文 ...

  4. Customize Spring @RequestParam Deserialization for Maps and/or Nested Objects

    @RestController class MyController { @RequestMapping(...) public void test(Container container) { .. ...

  5. 基于visual Studio2013解决面试题之0506取和为m的可能组合

     题目

  6. HTML语言简单回顾

    简单复习一下html语言. html的基本结构如下: <html> <head> <title></title> </head> <b ...

  7. jsp静态化之简单介绍

    package com.zwd.text; import java.io.BufferedReader; import java.io.File; import java.io.FileOutputS ...

  8. 单片机C语言实现的采用DS18B20的温度检测装置

    这几天老师布置了一个课程设计题目:采用51单片机控制的DS18B20温度检测系统.大概花了我一个礼拜的时间,幸好我的C语言学得还可以,最后还是让我搞出来了,真是高兴,我是采用STC-52单片机和DS1 ...

  9. 14.6.2 Moving or Copying InnoDB Tables to Another Machine 移动或者copy InnoDB 表到另外的机器

    14.6.2 Moving or Copying InnoDB Tables to Another Machine 移动或者copy InnoDB 表到另外的机器 这个章节描述技术关于移动或者copy ...

  10. 如果是在有master上开启了该参数,记得在slave端也要开启这个参数(salve需要stop后再重新start),否则在master上创建函数会导致replaction中断。

    如果是在有master上开启了该参数,记得在slave端也要开启这个参数(salve需要stop后再重新start),否则在master上创建函数会导致replaction中断.