Java ForkJoin框架的原理及用法

作者:ねぇ 时间:2022-03-13 05:26:51 

这篇文章主要介绍了Java ForkJoin框架的原理及用法,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下

ForkJoin分析

一、ForkJoin

ForkJoin是由JDK1.7后提供多线并发处理框架。ForkJoin的框架的基本思想是分而治之。什么是分而治之?分而治之就是将一个复杂的计算,按照设定的阈值进行分解成多个计算,然后将各个计算结果进行汇总。相应的ForkJoin将复杂的计算当做一个任务。而分解的多个计算则是当做一个子任务。

Java ForkJoin框架的原理及用法

二、ForkJoin的使用

下面我们以计算一个长度为一个亿的随机数整数数组为例来展示一下ForkJoin的使用:

1.创建Task

使用ForkJoin框架,需要创建一个ForkJoin的任务,而ForkJoinTask是一个抽象类,我们不需要去继承ForkJoinTask进行使用。因为ForkJoin框架为我们提供了RecursiveAction和RecursiveTask(子任务有返回值)。我们只需要继承ForkJoin为我们提供的抽象类的其中一个并且实现compute方法。


private static class SumTask extends RecursiveTask<Integer>{

//设定计算长度的阈值为总长度的十分之一也就是一千万
 private final static int THRESHOLD = MakeArray.ARRAY_LENGTH/10;
 private int[] src; //表示我们要实际统计的数组
 private int fromIndex;//开始统计的下标
 private int toIndex;//统计到哪里结束的下标

public SumTask(int[] src, int fromIndex, int toIndex) {
   this.src = src;
   this.fromIndex = fromIndex;
   this.toIndex = toIndex;
 }

@Override
 protected Integer compute() {
   //计算长度如果小于设定长度就不需要分解任务
   if(toIndex-fromIndex < THRESHOLD) {
     int count = 0;
     for(int i=fromIndex;i<=toIndex;i++) {
       count = count + src[i];
     }
     return count;
   }else {
     int mid = (fromIndex+toIndex)/2;
     SumTask left = new SumTask(src,fromIndex,mid);
     SumTask right = new SumTask(src,mid+1,toIndex);
     //将任务进行拆分
     invokeAll(left,right);
     //连接返回结果
     return left.join()+right.join();
   }
 }
}

2.使用ForkJoinPool进行执行

task要通过ForkJoinPool来执行,分割的子任务也会添加到当前工作线程的双端队列中,进入队列的头部。当一个工作线程中没有任务时,会从其他工作线程的队列尾部获取一个任务(工作窃取)。


public static void main(String[] args) {
 ForkJoinPool pool = new ForkJoinPool();
 int[] src = MakeArray.makeArray();
 SumTask innerFind = new SumTask(src,0,src.length-1);
 long start = System.currentTimeMillis();
 pool.invoke(innerFind);//同步调用
}

三、ForkJoin注意点

  • 使用ForkJoin将相同的计算任务通过多线程的进行执行。从而能提高数据的计算速度。在google的中的大数据处理框架mapreduce就通过类似ForkJoin的思想。通过多线程提高大数据的处理。但是我们需要注意:

  • 使用这种多线程带来的数据共享问题,在处理结果的合并的时候如果涉及到数据共享的问题,我们尽可能使用JDK为我们提供的并发容器。

  • ForkJoin也是通过多线程的方式进行处理任务。那么我们不得不考虑是否应该使用ForkJoin。因为当数据量不是特别大的时候,我们没有必要使用ForkJoin。因为多线程会涉及到上下文的切换。所以数据量不大的时候使用串行比使用多线程快。

来源:https://www.cnblogs.com/lee0527/p/11679953.html

标签:java,forkjoin,框架
0
投稿

猜你喜欢

  • Spring Boot多个定时任务阻塞问题的解决方法

    2023-09-20 11:43:36
  • C++实现堆排序实例介绍

    2022-06-05 12:33:54
  • C# IP地址与整数之间转换的具体方法

    2023-06-20 13:33:25
  • java操作excel表格详解

    2021-08-20 14:35:46
  • 如何把JAR发布到maven中央仓库的几种方法

    2022-11-08 11:07:06
  • Unity3D实现攻击范围检测

    2023-07-02 12:12:39
  • 利用java实现邮箱群发功能

    2021-07-11 21:55:23
  • android 实现ScrollView自动滚动的实例代码

    2021-11-20 03:00:27
  • 微信随机生成红包金额算法java版

    2023-07-27 16:01:39
  • C#关闭指定名字进程的方法

    2021-11-23 23:59:42
  • Android图片处理工具类BitmapUtils

    2022-12-28 05:23:39
  • sin(x)如何求解的java代码实现方法

    2022-04-17 16:22:07
  • Android实现取消GridView中Item选中时默认的背景色

    2023-02-26 05:53:30
  • MybatisPlus代码生成器含XML文件详解

    2023-10-25 21:51:26
  • springboot 按月分表的实现方式

    2023-11-25 00:03:47
  • C#多线程及同步示例简析

    2022-02-25 22:12:43
  • C#数组学习相关资料整理

    2022-12-03 08:02:54
  • java转换字符串编码格式的方法

    2022-02-12 04:30:24
  • springboot redis分布式锁代码实例

    2023-12-05 21:40:10
  • Java基于对象流实现银行系统

    2023-09-10 02:54:21
  • asp之家 软件编程 m.aspxhome.com