ForkJoinPool 學習示例

? ??

? ? ?在JAVA7之前,并行處理數據非常麻煩。第一,你得明確把包含數據的數據結構分成若干份。第二,你要將每個子部分分配給一個獨立的線程。第三,你要在恰當的時候對它們進行同步避免不希望的競爭條件,等待所有線程完成,最后把這些部分結果合并起來。在Java 7引入了分支/合并框架,讓這些操作更穩定、更不容易出錯。

? ? ?分支/合并框架的目的是以遞歸的方式將可以并行的任務拆分為更小的任務,然后將每個子任務的結果合并起來生成整體結果。要把子任務提交到ForkJoinPool必須創建RecursiveTask<R>的子類。需要實現它唯一的抽象方法 protected abstract R compute(); ?在這個方法中定義了將任務拆分成子任務的邏輯,以及無法拆分時生成單個子任務結果的邏輯。

?

?

? 計算1到10000000的和

?

/*** Desc:Fork/Join框架的目的是以遞歸方式將可以并行的任務拆分為更小的任務,然后將每個子任務的結果合并起來生成一個整體結果。* 要把任務提交到ForkJoinPool必須創建RecursiveTask<T> 的一個子類* * @author wei.zw* @since 2016年7月6日 下午9:27:56* @version v 0.1*/
public class ForkJoinSumCalculator extends RecursiveTask<Long> {/**  */private static final long serialVersionUID = -8013303660374621470L;private final long[] numbers;private final int start;private final int end;private static final long THRESHOLD = 1000;/*** @param numbers* @param start* @param end*/public ForkJoinSumCalculator(long[] numbers, int start, int end) {super();this.numbers = numbers;this.start = start;this.end = end;}/*** @param numbers*/public ForkJoinSumCalculator(long[] numbers) {super();this.numbers = numbers;this.start = 0;this.end = numbers.length;}/*** @see java.util.concurrent.RecursiveTask#compute()*/@Overrideprotected Long compute() {int length = end - start;if (length <= THRESHOLD) {return computeSequentially();}ForkJoinSumCalculator leftTask = new ForkJoinSumCalculator(numbers, start, start + length / 2);leftTask.fork();ForkJoinSumCalculator rightTask = new ForkJoinSumCalculator(numbers, start + length / 2, end);rightTask.fork();Long rightResult = 0L;try {rightResult = rightTask.get();} catch (Exception e) {}Long leftResult = leftTask.join();return leftResult + rightResult;}/*** * @return* @author wei.zw*/private Long computeSequentially() {long sum = 0;for (int i = start; i < end; i++) {sum += numbers[i];}return sum;}public static void main(String[] args) {long[] numbers = LongStream.rangeClosed(1, 10000000).toArray();long start = System.currentTimeMillis();System.out.println(new ForkJoinPool().invoke(new ForkJoinSumCalculator(numbers)) + " 耗時:"+ (System.currentTimeMillis() - start));}}

  

?結果是:50000005000000 耗時:37

?

優化后的

/*** @see java.util.concurrent.RecursiveTask#compute()*/@Overrideprotected Long compute() {int length = end - start;if (length <= THRESHOLD) {return computeSequentially();}ForkJoinSumCalculator leftTask = new ForkJoinSumCalculator(numbers, start, start + length / 2);leftTask.fork();ForkJoinSumCalculator rightTask = new ForkJoinSumCalculator(numbers, start + length / 2, end);Long rightResult = rightTask.compute();Long leftResult = leftTask.join();return leftResult + rightResult;}

  

?計算結果是:50000005000000 耗時:25

?

使用Fork/Join框架的最佳做法:

  • 對一個任務調用join方法會阻塞調用方,直到該任務作出結果。因此,又必須要在兩個子任務的計算都開始之后再調用它。
  • 不應該在RecursiveTask內部使用ForkJoinPool的invoke方法,應該直接調用compute或者fork方法
  • 對子任務調用fork方法可以將這個子任務排進ForkJoinPool。同時對左右兩邊的子任務都調用似乎很自然,但是這樣做的效率比直接對其中一個調用compute方法低。這樣做可以為其中一個子任務重用同一線程,從而避免在線程池中多分配一個任務造成的開銷。

看完了基本示例,在分析一下源碼;首先看一下RecursiveTask,通過名稱可以知道這是一個遞歸Task.源碼很簡單

public abstract class RecursiveTask<V> extends ForkJoinTask<V> {private static final long serialVersionUID = 5232453952276485270L;//計算結果V result;//抽象的計算方法protected abstract V compute();//獲取計算結果public final V getRawResult() {return result;}//設置計算結果protected final void setRawResult(V value) {result = value;}//執行計算protected final boolean exec() {result = compute();return true;}}

 RecursiveTask源碼看完以后,繼續分析ForkJoinTask

 

??

?

轉載于:https://www.cnblogs.com/wei-zw/p/8797732.html

本文來自互聯網用戶投稿,該文觀點僅代表作者本人,不代表本站立場。本站僅提供信息存儲空間服務,不擁有所有權,不承擔相關法律責任。
如若轉載,請注明出處:http://www.pswp.cn/news/457660.shtml
繁體地址,請注明出處:http://hk.pswp.cn/news/457660.shtml
英文地址,請注明出處:http://en.pswp.cn/news/457660.shtml

如若內容造成侵權/違法違規/事實不符,請聯系多彩編程網進行投訴反饋email:809451989@qq.com,一經查實,立即刪除!

相關文章

python深入_Python深入(上)

作者&#xff1a;Vamei 出處&#xff1a;http://www.cnblogs.com/vamei 歡迎轉載&#xff0c;也請保留這段聲明。謝謝&#xff01;到現在為止&#xff0c;Python學習已經可以告一段落。下面的部分&#xff0c;我想討論Python的高級語法和底層實現。這一部分的內容并不是使用Pyt…

06_JS對象

JS對象 創建對象 構造函數&#xff1a;var obj new Object()對象字面量&#xff1a;var obj {}; 不同&#xff1a;對象字面量可以在創建的同時為對象設置屬性&#xff0c;最后一個屬性后面不要加逗號 對象操作 添加屬性&#xff1a;對象名.新屬性名 值修改屬性&#xf…

BZOJ4107 : [Wf2015]Asteroids

首先將速度相減&#xff0c;變成A在動而B不動&#xff0c;若速度為0則顯然永遠不會相交。 枚舉A的每個點以及B的每條線段&#xff0c;計算這三個點共線的時刻。 將時刻排序&#xff0c;對于每個區間進行三分&#xff0c;用半平面交計算相交面積。 注意特判相交面積為0但是存在交…

Web Service 性能測試工具比較

背景 希望選擇一款Web Service性能測試工具&#xff0c;能真實模擬大量用戶訪問網站時的請求&#xff0c;從而獲取服務器當前的請求處理能力&#xff08;請求數/秒&#xff09;。以微信服務器為例&#xff0c;每個用戶用獨立的登錄token&#xff0c;做各種操作&#xff0c;比如…

python中的常量是什么意思_第14p,Python中的常量與注釋。

大家好&#xff0c;我是楊數Tos&#xff0c;這是《從零基礎到大神》系列課程的第14篇文章&#xff0c;第二階段的課程&#xff1a;Python基礎知識&#xff1a;常量與注釋。學習本課程&#xff0c;建議先看一遍&#xff1a;【計算機基礎知識】課程。一、Python中的常量1、什么是…

07_JS函數

JS函數 函數聲明 使用構造函數 var fun new Function();使用關鍵字 function 函數名(形參){// 函數體 }匿名函數 var fun1 function(){name "fun1" }函數參數 實參&#xff0c;形參都可以是任意數據類型&#xff0c;瀏覽器執行時不會檢查實參類型和數量&…

cocoapods的安裝(這真是一個神奇的東西,每次安裝的方法都不一樣,而且很容易出現各種各樣的錯誤)...

文章開始之前&#xff0c;建議安裝一個顯示網速的插件&#xff0c;不然你不知道到底有沒有下載&#xff0c;也讓生活有一點盼頭 1.因為眾所周知的原因&#xff08;我dang的行為真的是讓人失望&#xff09;&#xff0c;先更換一下ruby鏡像源 $ gem sources --remove https://rub…

轉】MyEclipse使用總結——在MyEclipse中設置jsp頁面為默認utf-8編碼

原博文出自于&#xff1a;http://www.cnblogs.com/xdp-gacl/p/3496161.html        感謝&#xff01; 在MyEclispe中創建Jsp頁面&#xff0c;Jsp頁面的默認編碼是“ISO-8859-1”&#xff0c;如下圖所示&#xff1a; 在這種編碼下編寫中文是沒有辦法保存Jsp頁面的&#…

sum怎么用python_python sum()函數和.sum(axis=0)函數的使用

參考&#xff1a;《Machine Learning in Action》第二章#####################################################################sum函數&#xff1a;sum()函數是內建函數help(sum)函數功能&#xff1a;返回一個數字序列(非字符串)的和&#xff0c;并加上參數start的值(默認為…

08_JS工廠方法和構造函數

工廠方法和構造函數 工廠方法 工廠方法用來批量產生對象 function CreatCar(name,prase){// new 一個對象var obj new Object();obj.name name;obj.parse prase;obj.print function(){console.log(this.name " : " this.parse)};// 返回對象return obj; };va…

華為機試題【10】-求數字基root

題目描述&#xff1a; 求整數的Root:給定正整數,求每位數字之和;如果和不是一位數,則重復; 輸入&#xff1a;輸入任意一個或多個整數 輸出&#xff1a;輸出各位數字之和,直到和為個位數為止(輸入異常,則返回-1)&#xff0c;多行&#xff0c;每行對應一個輸入數據的結果。 樣例…

國內開源鏡像站

搜狐開源鏡像站&#xff1a;http://mirrors.sohu.com/ 網易開源鏡像站&#xff1a;http://mirrors.163.com/開源中國&#xff1a;http://mirrors.oschina.net/首都在線科技股份有限公司&#xff1a;http://mirrors.yun-idc.com/阿里云開源鏡像&#xff1a;http://mirrors.aliyu…

python update skeleton 不自動_python編程筆記(1)-數據類型

斷斷續續學了一些&#xff0c;但是后邊一直沒有使用&#xff0c;就又忘記了。為了縮短學習周期&#xff0c;這次學習的方法&#xff1a;先學主要框架&#xff08;數據類型、條件、循環、函數、類、numpy、pandas、sklearn&#xff09;&#xff0c;后期直接看朋友code 實操了&am…

09_JS原型對象

原型對象 在構造函數中&#xff0c;同一類相同的方法可以直接寫在構造函數里&#xff0c;但這樣每實例化一個對象都會產生一個新的該方法&#xff0c;但其實這個方法都是一樣的&#xff0c;這樣會浪費大量空間&#xff0c;另外&#xff0c;可以將相同的方法寫在構造函數外&…

[改善Java代碼]非穩定排序推薦使用List

我們知道Set與List的最大區別就是Set中的元素不可以重復&#xff08;這個重復指的equals方法的返回值相等&#xff09;&#xff0c;其他方面則沒有太大的區別了&#xff0c;在Set的實現類中有一個比較常用的類需要了解一下&#xff1a;TreeSet&#xff0c;該類實現了類默認排序…

如何訪問另一臺電腦的共享文件夾_如何遠程控制另一臺電腦

在工作中&#xff0c;我們時常需要遠程控制一下另一臺電腦&#xff0c;如果這兩臺計算機在局域網內可以通過遠程桌面輕松實現&#xff0c;如果這兩臺計算機不在局域網內則通常需要借助第三方軟件來實現。遠程桌面使用分三步&#xff1a;1、被控制端要開啟遠程桌面功能&#xff…

10_JS數組

JS 數組 以索引為鍵的對象&#xff0c;性能高于對象 創建 new Array() var arr Array() for(var i 0;i<5;i){arr[i] i; };var arr Array(1,2,3,4,5,6)數組字面量 var arr [2,3,4,5,6,7,8];不同&#xff1a;傳入一個整數時&#xff0c;new把這個數當作數組長度&…

微軟Hololens學院教程- Holograms 101: Introduction with Device【微軟教程已經更新,本文是老版本】...

這是老版本的教程&#xff0c;為了不耽誤大家的時間&#xff0c;請直接看原文&#xff0c;本文僅供參考哦&#xff01;原文鏈接&#xff1a;https://developer.microsoft.com/EN-US/WINDOWS/HOLOGRAPHIC/holograms_101 這篇文章將通過一個完整的實例來了解設備的核心特性&#…

python seaborn boxplot_python - 如何將文本框添加到Seaborn Boxplot? - 堆棧內存溢出

我試圖在框圖中插入一個文本框&#xff0c;但是出了點問題。 讓我們先繪制箱線圖&#xff1a;import pandas as pdimport matplotlib.pyplot as pltimport seaborn as sbdf_data pd.read_hdf(data.h5, keyfdf_data)##print(df_data)fig, (ax1, ax2) plt.subplots(1,2,gridspe…

JS_11正則表達式和字符串方法

正則表達式和字符串方法 正則表達式 創建正則對象 構造函數字面量 匹配模式 i&#xff1a;忽略大小寫m&#xff1a;多行匹配g&#xff1a;全局匹配 正則語法 正則作用/a/檢查是否有a/a|b/檢查是否有a或b{}量詞&#xff0c;如{1,3}匹配1到3次&#xff0c;{m,}匹配m次以上…