日本不卡不码高清免费观看,久久国产精品久久w女人spa,黄色aa久久,三上悠亚国产精品一区二区三区

您的位置:首頁技術文章
文章詳情頁

Java并發中的Fork/Join 框架機制詳解

瀏覽:40日期:2022-08-08 17:21:52
什么是 Fork/Join 框架

Fork/Join 框架是一種在 JDk 7 引入的線程池,用于并行執行把一個大任務拆成多個小任務并行執行,最終匯總每個小任務結果得到大任務結果的特殊任務。通過其命名也很容易看出框架主要分為 Fork 和 Join 兩個階段,第一階段 Fork 是把一個大任務拆分為多個子任務并行的執行,第二階段 Join 是合并這些子任務的所有執行結果,最后得到大任務的結果。

這里不難發現其執行主要流程:首先判斷一個任務是否足夠小,如果任務足夠小,則直接計算,否則,就拆分成幾個更小的小任務分別計算,這個過程可以反復的拆分成一系列小任務。Fork/Join 框架是一種基于 分治 的算法,通過拆分大任務成多個獨立的小任務,然后并行執行這些小任務,最后合并小任務的結果得到大任務的最終結果,通過并行計算以提高效率。。

Fork/Join 框架使用示例

下面通過一個計算列表中所有元素的總和的示例來看看 Fork/Join 框架是如何使用的,總的思路是:將這個列表分成許多子列表,然后對每個子列表的元素進行求和,然后,我們再計算所有這些值的總和就得到原始列表的和了。Fork/Join 框架中定義了 ForkJoinTask 來表示一個 Fork/Join 任務,其提供了 fork()、join() 等操作,通常情況下,我們并不需要直接繼承這個 ForkJoinTask 類,而是使用框架提供的兩個 ForkJoinTask 的子類:

RecursiveAction 用于表示沒有返回結果的 Fork/Join 任務。 RecursiveTask 用于表示有返回結果的 Fork/Join 任務。

很顯然,在這個示例中是需要返回結果的,可以定義 SumAction 類繼承自 RecursiveTask,代碼入下:

/** * @author mghio * @since 2021-07-25 */public class SumTask extends RecursiveTask<Long> { private static final int SEQUENTIAL_THRESHOLD = 50; private final List<Long> data; public SumTask(List<Long> data) { this.data = data; } @Override protected Long compute() { if (data.size() <= SEQUENTIAL_THRESHOLD) { long sum = computeSumDirectly(); System.out.format('Sum of %s: %dn', data.toString(), sum); return sum; } else { int mid = data.size() / 2; SumTask firstSubtask = new SumTask(data.subList(0, mid)); SumTask secondSubtask = new SumTask(data.subList(mid, data.size())); // 執行子任務 firstSubtask.fork(); secondSubtask.fork(); // 等待子任務執行完成,并獲取結果 long firstSubTaskResult = firstSubtask.join(); long secondSubTaskResult = secondSubtask.join(); return firstSubTaskResult + secondSubTaskResult; } } private long computeSumDirectly() { long sum = 0; for (Long l : data) { sum += l; } return sum; } public static void main(String[] args) { Random random = new Random(); List<Long> data = random.longs(1_000, 1, 100).boxed().collect(Collectors.toList()); ForkJoinPool pool = new ForkJoinPool(); SumTask task = new SumTask(data); pool.invoke(task); System.out.println('Sum: ' + pool.invoke(task)); }}

這里當列表大小小于 SEQUENTIAL_THRESHOLD 變量的值(閾值)時視為小任務,直接計算求和列表元素結果,否則再次拆分為小任務,運行結果如下:

Java并發中的Fork/Join 框架機制詳解

通過這個示例代碼可以發現,Fork/Join 框架 中 ForkJoinTask 任務與平常的一般任務的主要不同點在于:ForkJoinTask 需要實現抽象方法 compute() 來定義計算邏輯,在這個方法里一般通用的實現模板是,首先先判斷當前任務是否是小任務,如果是,就執行執行任務,如果不是小任務,則再次拆分為兩個子任務,然后當每個子任務調用 fork() 方法時,會再次進入到 compute() 方法中,檢查當前任務是否需要再拆分為子任務,如果已經是小任務,則執行當前任務并返回結果,否則繼續分割,最后調用 join() 方法等待所有子任務執行完成并獲得執行結果。偽代碼如下:

if (problem is small) { directly solve problem.} else { Step 1. split problem into independent parts. Step 2. fork new subtasks to solve each part. Step 3. join all subtasks. Step 4. compose result from subresults.}Fork/Join 框架設計

Fork/Join 框架核心思想是把一個大任務拆分成若干個小任務,然后匯總每個小任務的結果最終得到大任務的結果,如果讓你設計一個這樣的框架,你會如何實現呢?(建議思考一下),Fork/Join 框架的整個流程正如其名所示,分為兩個步驟:

大任務分割 需要有這么一個的類,用來將大任務拆分為子任務,可能一次拆分后的子任務還是比較大,需要多次拆分,直到拆分出來的子任務符合我們定義的小任務才結束。 執行任務并合并任務結果 第一步拆分出來的子任務分別存放在一個個 雙端隊列 里面(P.S. 這里為什么要使用雙端隊列請看下文),然后每個隊列啟動一個線程從隊列中獲取任務執行。這些子任務的執行結果都會放到一個統一的隊列中,然后再啟動一個線程從這個隊列中拿數據,最后合并這些數據返回。

Fork/Join 框架使用了如下兩個類來完成以上兩個步驟:

ForkJoinTask 類 在上文的實例中也有提到,表示 ForkJoin 任務,在使用框架時首先必須先定義任務,通常只需要繼承自 ForkJoinTask 類的子類 RecursiveAction(無返回結果) 或者 RecursiveTask(有返回結果)即可。 ForkJoinPool 從名字也可以猜到一二了,就是用來執行 ForkJoinTask 的線程池。大任務拆分出的子任務會添加到當前線程的雙端隊列的頭部。

喜歡思考的你,心中想必會想到這么一種場景,當我們需要完成一個大任務時,會先把這個大任務拆分為多個獨立的子任務,這些子任務會放到獨立的隊列中,并為每個隊列都創建一個單獨的線程去執行隊列里的任務,即這里線程和隊列時一對一的關系,那么當有的線程可能會先把自己隊列的任務執行完成了,而有的線程則沒有執行完成,這就導致一些先執行完任務的線程干等了,這是個好問題。

既然是做并發的,肯定要最大程度壓榨計算機的性能,對于這種場景并發大師 Doug Lea 使用了工作竊取算法處理,使用工作竊取算法后,先完成自己隊列中任務的線程會去其它線程的隊列中”竊取“一個任務來執行,哈哈,一方有難,八方支援。但是此時這個線程和隊列的持有線程會同時訪問同一個隊列,所以為了減少竊取任務的線程和被竊取任務的線程之間的競爭,ForkJoin 選擇了雙端隊列這種數據結構,這樣就可以按照這種規則執行任務了:被竊取任務的線程始終從隊列頭部獲取任務并執行,竊取任務的線程使用從隊列尾部獲取任務執行。這個算法在絕大部分情況下都可以充分利用多線程進行并行計算,但是在雙端隊列里只有一個任務等極端情況下還是會存在一定程度的競爭。

Java并發中的Fork/Join 框架機制詳解

Fork/Join 框架實現原理

Fork/Join 框架的實現核心是 ForkJoinPool 類,該類的重要組成部分為 ForkJoinTask 數組和 ForkJoinWorkerThread 數組,其中 ForkJoinTask 數組用來存放框架使用者給提交給 ForkJoinPool 的任務,ForkJoinWorkerThread 數組則負責執行這些任務。任務有如下四種狀態:

NORMAL 已完成

CANCELLED 被取消

SIGNAL 信號

EXCEPTIONAL 發生異常

下面來看看這兩個類的核心方法實現原理,首先來看 ForkJoinTask 的 fork() 方法,源碼如下:

Java并發中的Fork/Join 框架機制詳解

方法對于 ForkJoinWorkerThread 類型的線程,首先會調用 ForkJoinWorkerThread 的 workQueue 的 push() 方法異步的去執行這個任務,然后馬上返回結果。繼續跟進 ForkJoinPool 的 push() 方法,源碼如下:

Java并發中的Fork/Join 框架機制詳解

方法將當前任務添加到 ForkJoinTask 任務隊列數組中,然后再調用 ForkJoinPool 的 signalWork 方法創建或者喚醒一個工作線程來執行該任務。然后再來看看 ForkJoinTask 的 join() 方法,方法源碼如下:

Java并發中的Fork/Join 框架機制詳解

Java并發中的Fork/Join 框架機制詳解

方法首先調用了 doJoin() 方法,該方法返回當前任務的狀態,根據返回的任務狀態做不同的處理:

已完成狀態則直接返回結果 被取消狀態則直接拋出異常(CancellationException) 發生異常狀態則直接拋出對應的異常

繼續跟進 doJoin() 方法,方法源碼如下:

Java并發中的Fork/Join 框架機制詳解

方法首先判斷當前任務狀態是否已經執行完成,如果執行完成則直接返回任務狀態。如果沒有執行完成,則從任務數組中(workQueue)取出任務并執行,任務執行完成則設置任務狀態為 NORMAL,如果出現異常則記錄異常并設置任務狀態為 EXCEPTIONAL(在 doExec() 方法中)。

總結

本文主要介紹了 Java 并發框架中的 Fork/Join 框架的基本原理和其使用的工作竊取算法(work-stealing)、設計方式和部分實現源碼。Fork/Join 框架在 JDK 的官方標準庫中也有應用。比如 JDK 1.8+ 標準庫提供的 Arrays.parallelSort(array) 可以進行并行排序,它的原理就是內部通過 Fork/Join 框架對大數組分拆進行并行排序,可以提高排序的速度,還有集合中的 Collection.parallelStream() 方法底層也是基于 Fork/Join 框架實現的,最后就是定義小任務的閾值往往是需要通過測試驗證才能合理給出,并且保證程序可以達到最好的性能。

到此這篇關于Java 并發中的Fork/Join 框架機制詳解的文章就介紹到這了,更多相關Java Fork/Join 框架內容請搜索好吧啦網以前的文章或繼續瀏覽下面的相關文章希望大家以后多多支持好吧啦網!

標簽: Java
相關文章:
日本不卡不码高清免费观看,久久国产精品久久w女人spa,黄色aa久久,三上悠亚国产精品一区二区三区
日韩欧美中文字幕一区二区三区 | www.九色在线| 久久久免费人体| 久久爱www成人| 久久婷婷国产| 国产福利电影在线播放| 国产精品久久久久av电视剧| 99精品视频在线| 红桃视频欧美| 亚洲一区有码| 欧美亚洲专区| 欧美激情精品| 麻豆91小视频| 日韩影院二区| 亚洲女同一区| 亚洲一二三区视频| 国产色99精品9i| 精品久久电影| 日韩精品一区二区三区免费观影| 欧美一区二区性| 日韩在线a电影| 国产视频一区二| 国产在线一区不卡| 99久久视频| 只有精品亚洲| 国产精品久一| 日韩网站中文字幕| 99热国内精品| 视频一区二区三区入口| 91精品国产自产观看在线| 久久久久观看| 亚洲国产日韩欧美在线| 日韩高清一区二区| 老牛国内精品亚洲成av人片| 成人午夜国产| 日韩精品一二区| 国产欧美大片| 四虎884aa成人精品最新| 亚洲综合不卡| 国产精品22p| 亚洲欧美一区在线| 日韩三级一区| 国产一区二区三区四区大秀| 好吊一区二区三区| 国产伦一区二区三区| 欧美aa在线观看| 免费观看在线综合| 成人国产综合| 亚洲综合图色| 高潮久久久久久久久久久久久久| 不卡一区2区| 欧美久久一区二区三区| 久久高清免费| 欧美另类中文字幕| 久久久久国产| 日韩成人一级| 欧美/亚洲一区| 青青在线精品| 婷婷亚洲五月色综合| 国产精品分类| 99亚洲视频| 激情不卡一区二区三区视频在线| 天堂av在线一区| 高清久久精品| 日本一区免费网站| 免费污视频在线一区| 日韩黄色在线观看| 欧美日韩一区二区三区视频播放| 国产毛片一区二区三区| 91国语精品自产拍| 精品日产乱码久久久久久仙踪林| 麻豆亚洲精品| 日韩.com| 久久激情综合网| 在线亚洲激情| av高清不卡| 国产欧美一区二区精品久久久 | 亚洲制服一区| www成人在线视频| 久久狠狠亚洲综合| 久久一区二区三区喷水| 欧美日韩一区二区三区在线电影| 欧美理论视频| 日韩av在线播放网址| 日韩精品一区二区三区av| 国产一区日韩一区| 高潮久久久久久久久久久久久久| 日韩福利在线观看| 亚洲中字黄色| 亚洲大片在线| 亚洲一区资源| 开心激情综合| 久久国产精品免费精品3p| 中文不卡在线| 红桃视频国产精品| 色综合狠狠操| 久久a爱视频| 国产欧美一区| 日韩va欧美va亚洲va久久| 亚洲综合丁香| 欧美日韩国产高清电影| 色婷婷久久久| 最新中文字幕在线播放 | 牛牛精品成人免费视频| 日本免费新一区视频| 亚洲一区二区三区中文字幕在线观看| 久久精品亚洲人成影院| 四虎8848精品成人免费网站| 国产精品2023| 欧美一级网站| 欧美在线精品一区| 日韩激情啪啪| 日韩av网站在线免费观看| 亚洲三级国产| 老牛影视一区二区三区| 国产视频久久| 西西人体一区二区| 国产精品日本欧美一区二区三区| re久久精品视频| 婷婷亚洲综合| 日韩一级不卡| 久久国产高清| 亚洲日产国产精品| 日日夜夜免费精品| 久久国产66| 日韩专区一卡二卡| 久久福利毛片| 亚洲欧美网站在线观看| 四虎精品一区二区免费| 日本中文字幕不卡| 欧美一区91| 免费日韩成人| 日韩av在线中文字幕| 久久裸体视频| 性欧美69xoxoxoxo| 久久亚洲视频| 日本亚洲欧美天堂免费| 啪啪亚洲精品| 久久精品亚洲一区二区| 久久成人高清| 国产美女高潮在线| 久久久精品久久久久久96| 激情婷婷综合| 亚洲影视一区二区三区| 国产亚洲欧美日韩精品一区二区三区| 麻豆精品一区二区综合av| 伊人久久视频| 影音先锋国产精品| 日韩成人在线看| 老司机精品视频网| 久久精品91| 奶水喷射视频一区| 日本色综合中文字幕| 精品国产鲁一鲁****| 国产精品久久久久久久免费观看 | 日韩精品一级二级| 国产欧美亚洲一区| 岛国av在线播放| 国产精品婷婷| 国产日韩高清一区二区三区在线| 国产va免费精品观看精品视频| 亚洲二区视频| 日韩高清中文字幕一区| 久久精品亚洲| 激情视频一区二区三区| 亚洲一区av| 卡一精品卡二卡三网站乱码| 在线成人动漫av| 亚洲精品伊人| 日本一区二区免费高清| 午夜亚洲一区| 毛片不卡一区二区| 黄色av一区| 国产精品亚洲欧美| 日韩精品免费一区二区在线观看| 视频在线观看91| 精品国产欧美日韩| 国产精品老牛| 精品伊人久久| 最新日韩欧美| 麻豆精品av| 日韩制服丝袜av| 97国产成人高清在线观看| 蜜桃视频一区二区| 国产精品成人a在线观看| 免费不卡在线观看| 久久精品国内一区二区三区| 一本色道久久精品| 精品美女久久| 亚洲香蕉久久| 婷婷综合六月| 国产欧美一区二区三区精品观看| 激情五月色综合国产精品| 国产精品观看| 午夜一区在线| 日韩伦理一区| 国产精品成人一区二区网站软件| 中文国产一区| 日韩国产欧美一区二区| 日本a级不卡|