什么是Fork/Join框架
Fork/Join框架是Java7提供了的一个用于并行执行任务的框架,采用类似于分治算法,就是把一个复杂的问题分成两个或更多的相同或相似的子问题,直到最后子问题可以简单的直接求解,原问题的解即子问题的解的合并。
在这个框架中值得注意的一个重要概念是在理想状态下是没有空闲的工作线程。 它们实现了一种工作窃取算法,闲置的工作线程可以从忙碌的工作线程拿工作执行。
Fork/Join框架处理复杂的线程问题,你只需向框架指出哪些部分工作可以分解并递归处理。伪代码来自Doug Lea's的论文
Result solve(Problem problem) {
if (problem is small)
directly solve problem
else {
split problem into independent parts
fork new subtasks to solve each part
join all subtasks
compose result from subresults
}
}
Fork/Join框架的核心类
ForkJoinPool和ForkJoinTask是支持Fork/Join机制的核心类。
ForkJoinPool
ForkjoinPool是实现了ExecutorService和work-stealing(工作窃取)算法。如下所示,新建ForkJoinPool实例,指定并行等级(处理器的个数)。
ForkJoinPool pool = new ForkJoinPool(numberOfProcessors);
Where numberOfProcessors = Runtime.getRunTime().availableProcessors();
如果使用无参构造函数,默认创建pool的大小为上面所示的可用的处理器个数。尽管你可以指定任意大小的pool,但pool会动态调整大小来尝试获得足够的活动线程。与ExecutorService另一个重要的不同,pool不需要在程序退出时显式关闭,因为它的所有线程都处于守护进程模式。
三种提交任务到ForkJoinPool的方法:
- execute():期望异步执行,调用其fork方法在多个线程之间拆分工作。
- invoke():等待获得结果。
- submit():完成时返回一个future对象用于检查状态以及运行结果。
ForkJoinTask
ForkJoinTask是在ForkJoinPool创建工作的抽象类,RecursiveAction 和RecursiveTask是ForkJoinTask的直接子类,都要实现compute方法,两者唯一的不同点是:RecursiveAction没有返回任务的结果,而 RecursiveTask有返回任务的结果(可以自己指定类型的对象)。
ForkJoinTask类提供了几个方法用于检查任务运行的状态. 无论以什么方式结束任务,isDone() 方法返回true;如果完成任务过程中没有被取消或者发生异常,CompletedNormally() 方法返回true;如果任务被取消, isCancelled() 方法返回true;如果任务被取消或者遇到异常,isCompletedabnormally() 方法返回true。
异常处理代码如下:
if(task.isCompletedAbnormally())
{
System.out.println(task.getException());
}
getException方法返回Throwable对象,如果任务被取消了则返回CancellationException。如果任务没有完成或者没有抛出异常则返回null。
工作窃取算法
工作窃取(work-stealing)算法是指某个线程从其他队列里窃取任务来执行。假如我们需要做一个比较大的任务,我们可以把这个任务分割为若干互不依赖的子任务,为了减少线程间的竞争,于是把这些子任务分别放到不同的队列里,并为每个队列创建一个单独的线程来执行队列里的任务,线程和队列一一对应,比如A线程负责处理A队列里的任务。但是有的线程会先把自己队列里的任务干完,而其他线程对应的队列里还有任务等待处理。干完活的线程与其等着,不如去帮其他线程干活,于是它就去其他线程的队列里窃取一个任务来执行。而在这时它们会访问同一个队列,所以为了减少窃取任务线程和被窃取任务线程之间的竞争,通常会使用双端队列,被窃取任务线程永远从双端队列的头部拿任务执行,而窃取任务的线程永远从双端队列的尾部拿任务执行。
工作窃取算法的优点是充分利用线程进行并行计算,并减少了线程间的竞争,其缺点是在某些情况下还是存在竞争,比如双端队列里只有一个任务时。并且消耗了更多的系统资源,比如创建多个线程和多个双端队列。
Fork/Join框架和ExecutorService的区别
Fork/Join框架和ExecutorService最主要的区别是工作窃取算法。与Executor框架不同,当有线程完成了自己的所有子任务,而其他正在执行的线程(称为工作线程)还有子任务等待处理,就去其他线程的队列里窃取一个任务来执行。通过这种方式,线程可以充分利用其运行时间,从而提高应用程序的性能。
Fork/Join框架实践例子
在这个例子中,我们使用ForkJoinPool和ForkJoinTask提供的异步方法来管理任务。我们将实现遍历文件夹查找指定扩展名的文件,ForkJoinTask实现处理一个文件夹内的查找,如果存在子文件夹,为每一个文件夹fork一个新异步任务到ForkJoinPool中去,每个子任务会查找自己文件夹的指定扩展名的文件。一旦任务已经处理了所有的指定文件夹的内容,利用ForkJoinPool的join()方法等待完成所有任务。join方法是等待执行完成并返回compute()方法的计算结果。任务组的所有任务,都将自己的结果返回添加到结果列表中。
详细代码如下:
import java.io.File;
import java.util.ArrayList;
import java.util.List;
import java.util.concurrent.ForkJoinPool;
import java.util.concurrent.RecursiveTask;
import java.util.concurrent.TimeUnit;
public class FolderProcessor extends RecursiveTask> {
private final String path;
private final String extension;
public FolderProcessor(String path,String extension) {
this.extension = extension;
this.path = path;
}
@Override
protected List compute() {
List list = new ArrayList();
List tasks = new ArrayList();
File file = new File(path);
File[] content= file.listFiles();
if(content != null){
for (File aContent : content) {
if (aContent.isDirectory()) {
FolderProcessor task =
new FolderProcessor(aContent.getAbsolutePath(), extension);
task.fork();
tasks.add(task);
} else {
if (checkFile(aContent.getName())) {
list.add(aContent.getAbsolutePath());
}
}
}
}
if (tasks.size() > 50)
{
System.out.printf("%s: %d tasks ran.\n", file.getAbsolutePath(), tasks.size());
}
addResultsFromTasks(list, tasks);
return list;
}
private void addResultsFromTasks(List list, List tasks) {
for (FolderProcessor item : tasks)
{
list.addAll(item.join());
}
}
private boolean checkFile(String name) {
return name.endsWith(extension);
}
public static void main(String[] args) {
ForkJoinPool pool = new ForkJoinPool();
FolderProcessor system = new FolderProcessor("/System", "log");
FolderProcessor library = new FolderProcessor("/Library", "log");
FolderProcessor users = new FolderProcessor("/Users", "log");
pool.execute(system);
pool.execute(library);
pool.execute(users);
do
{
System.out.printf("******************************************\n");
System.out.printf("Main: Parallelism: %d\n", pool.getParallelism());
System.out.printf("Main: Active Threads: %d\n", pool.getActiveThreadCount());
System.out.printf("Main: Task Count: %d\n", pool.getQueuedTaskCount());
System.out.printf("Main: Steal Count: %d\n", pool.getStealCount());
System.out.printf("******************************************\n");
try
{
TimeUnit.SECONDS.sleep(1);
} catch (InterruptedException e)
{
e.printStackTrace();
}
} while ((!system.isDone()) || (!library.isDone()) || (!users.isDone()));
pool.shutdown();
List results;
results = system.join();
System.out.printf("System: %d files found.\n", results.size());
results = library.join();
System.out.printf("Library: %d files found.\n", results.size());
results = users.join();
System.out.printf("Users: %d files found.\n", results.size());
}
}
结果输出类似如下:
******************************************
Main: Parallelism: 8
Main: Active Threads: 60
Main: Task Count: 62370
Main: Steal Count: 81261
******************************************
******************************************
Main: Parallelism: 8
Main: Active Threads: 0
Main: Task Count: 19295
Main: Steal Count: 160629
******************************************
JDK中的使用实现
Java SE中有一些通用的功能,它们已经使用fork/join框架来实现。
1.在Java 8的java.util.Arrays中的parallelSort方法采用了fork/join框架,在多处理器系统上,并行排序大量数据比顺序排序更快
2.在Stream.parallel()中使用并行,更多请参考parallel stream operation in java 8。
总结:
设计优秀的多线程算法是非常困难的,fork/join框架并不适用于所有情况,但是在它的适用范围之内,能够轻松的利用多个CPU提供的计算资源来协作完成一个复杂的计算任务。最终还是看你的问题是否符合框架特性,若不符合,你可以使用基于java.util.concurrent包基础工具方法实现自己的解决方案。
参考
- Fork/Join Framework Tutorial: ForkJoinPool Example
- 聊聊并发(八)——Fork/Join框架介绍