Java ForkJoin框架的原理及用法
這篇文章主要介紹了Java ForkJoin框架的原理及用法,文中通過示例代碼介紹的非常詳細,對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友可以參考下
ForkJoin分析
一、ForkJoin
ForkJoin是由JDK1.7后提供多線并發(fā)處理框架。ForkJoin的框架的基本思想是分而治之。什么是分而治之?分而治之就是將一個復(fù)雜的計算,按照設(shè)定的閾值進行分解成多個計算,然后將各個計算結(jié)果進行匯總。相應(yīng)的ForkJoin將復(fù)雜的計算當做一個任務(wù)。而分解的多個計算則是當做一個子任務(wù)。

二、ForkJoin的使用
下面我們以計算一個長度為一個億的隨機數(shù)整數(shù)數(shù)組為例來展示一下ForkJoin的使用:
1.創(chuàng)建Task
使用ForkJoin框架,需要創(chuàng)建一個ForkJoin的任務(wù),而ForkJoinTask是一個抽象類,我們不需要去繼承ForkJoinTask進行使用。因為ForkJoin框架為我們提供了RecursiveAction和RecursiveTask(子任務(wù)有返回值)。我們只需要繼承ForkJoin為我們提供的抽象類的其中一個并且實現(xiàn)compute方法。
private static class SumTask extends RecursiveTask<Integer>{
//設(shè)定計算長度的閾值為總長度的十分之一也就是一千萬
private final static int THRESHOLD = MakeArray.ARRAY_LENGTH/10;
private int[] src; //表示我們要實際統(tǒng)計的數(shù)組
private int fromIndex;//開始統(tǒng)計的下標
private int toIndex;//統(tǒng)計到哪里結(jié)束的下標
public SumTask(int[] src, int fromIndex, int toIndex) {
this.src = src;
this.fromIndex = fromIndex;
this.toIndex = toIndex;
}
@Override
protected Integer compute() {
//計算長度如果小于設(shè)定長度就不需要分解任務(wù)
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);
//將任務(wù)進行拆分
invokeAll(left,right);
//連接返回結(jié)果
return left.join()+right.join();
}
}
}
2.使用ForkJoinPool進行執(zhí)行
task要通過ForkJoinPool來執(zhí)行,分割的子任務(wù)也會添加到當前工作線程的雙端隊列中,進入隊列的頭部。當一個工作線程中沒有任務(wù)時,會從其他工作線程的隊列尾部獲取一個任務(wù)(工作竊取)。
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);//同步調(diào)用
}
三、ForkJoin注意點
- 使用ForkJoin將相同的計算任務(wù)通過多線程的進行執(zhí)行。從而能提高數(shù)據(jù)的計算速度。在google的中的大數(shù)據(jù)處理框架mapreduce就通過類似ForkJoin的思想。通過多線程提高大數(shù)據(jù)的處理。但是我們需要注意:
- 使用這種多線程帶來的數(shù)據(jù)共享問題,在處理結(jié)果的合并的時候如果涉及到數(shù)據(jù)共享的問題,我們盡可能使用JDK為我們提供的并發(fā)容器。
- ForkJoin也是通過多線程的方式進行處理任務(wù)。那么我們不得不考慮是否應(yīng)該使用ForkJoin。因為當數(shù)據(jù)量不是特別大的時候,我們沒有必要使用ForkJoin。因為多線程會涉及到上下文的切換。所以數(shù)據(jù)量不大的時候使用串行比使用多線程快。
以上就是本文的全部內(nèi)容,希望對大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
Springboot實現(xiàn)阿里云通信短信服務(wù)有關(guān)短信驗證碼的發(fā)送功能
短信驗證碼是通過發(fā)送驗證碼到手機的一種有效的驗證碼系統(tǒng)。主要用于驗證用戶手機的合法性及敏感操作的身份驗證。下面通過本文大家分享Springboot實現(xiàn)阿里云通信短信服務(wù)有關(guān)短信驗證碼的發(fā)送功能,一起看看吧2017-08-08
java實現(xiàn)解析json復(fù)雜數(shù)據(jù)的方法詳解
這篇文章主要為大家詳細介紹了java如何實現(xiàn)解析json復(fù)雜數(shù)據(jù),文中的示例代碼講解詳細,具有一定的借鑒價值,感興趣的小伙伴可以學(xué)習(xí)一下2024-01-01
SpringBoot整合mybatis-generator-maven-plugin的方法
這篇文章主要介紹了SpringBoot整合mybatis-generator-maven-plugin,本文通過實例代碼給大家介紹的非常詳細,對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友可以參考下2020-11-11
springboot如何開啟緩存@EnableCaching(使用redis)
在Spring Boot項目中集成Redis主要包括添加依賴到pom.xml、配置application.yml中的Redis連接參數(shù)、編寫配置類、在啟動類上添加@EnableCaching注解以及測試接口的查詢和緩存驗證等步驟,首先,需要在pom.xml中添加spring-boot-starter-data-redis依賴2024-11-11
Java簡單實現(xiàn)對一串數(shù)字采用相應(yīng)的加密策略后傳輸
下面小編就為大家?guī)硪黄狫ava簡單實現(xiàn)對一串數(shù)字采用相應(yīng)的加密策略后傳輸。小編覺得挺不錯的,現(xiàn)在就分享給大家,也給大家做個參考。一起跟隨小編過來看看吧2016-09-09
深入分析Android系統(tǒng)中SparseArray的源碼
這篇文章主要介紹了深入分析Android系統(tǒng)中SparseArray的源碼,SparseArray為Java實現(xiàn),需要的朋友可以參考下2015-07-07
Java獲取網(wǎng)絡(luò)文件并插入數(shù)據(jù)庫的代碼
抓取各大網(wǎng)站的數(shù)據(jù)插入數(shù)據(jù)庫,這樣就不用為沒有數(shù)據(jù)而煩惱了2010-06-06
java中ThreadLocal的應(yīng)用場景實例分析
在本篇文章里小編給大家整理的是一篇關(guān)于java中ThreadLocal的應(yīng)用場景實例分析,對此有興趣的朋友們可以學(xué)習(xí)參考下。2021-02-02
spring中BeanUtils.copyProperties的使用(深拷貝,淺拷貝)
本文主要介紹了spring中BeanUtils.copyProperties的使用(深拷貝,淺拷貝),文中通過示例代碼介紹的非常詳細,對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2023-05-05

