激情久久久_欧美视频区_成人av免费_不卡视频一二三区_欧美精品在欧美一区二区少妇_欧美一区二区三区的

服務器之家:專注于服務器技術及軟件下載分享
分類導航

PHP教程|ASP.NET教程|JAVA教程|ASP教程|

服務器之家 - 編程語言 - JAVA教程 - Java Fork/Join框架

Java Fork/Join框架

2020-06-15 11:17robin JAVA教程

Fork/Join框架是Java7中新增的一項特性,也是Java7平臺的其中一項主要改進。下面我們就來簡單探討下Java的Fork/Join框架

Fork/Join框架是ExecutorService接口的一個實現,通過它我們可以實現多進程。Fork/Join可以用來將一個大任務遞歸的拆分為多個小任務,目標是充分利用所有的資源盡可能增強應用的性能。

和任何ExecutorService接口的實現一樣,Fork/Join也會使用線程池來分布式的管理工作線程。Fork/Join框架的獨特之處在于它使用了work-stealing(工作竊取)算法。通過這個算法,工作線程在無事可做時可以竊取其它正在繁忙的線程的任務來執行。

Fork/Join框架的核心是ForkJoinPool類,一個AbstractExecutorService類的子類。ForkJoinPool實現了核心的work-stealing算法并可以執行ForkJoinTask處理。

基礎用法

使用Fork/Join框架的第一步是編寫執行碎片任務的代碼。要編寫的代碼類似如下偽代碼:

?
1
2
3
4
5
if 任務足夠小:
 直接執行任務
else:
 將任務切成兩個小任務
 執行兩個小任務并等待結果

使用ForkJoinTask子類來封裝如上的代碼,通常會使用一些JDK提供的類,使用的有RecursiveTask(這個類會返回一個結果)和RecursiveAction兩個類。

在準備好ForkJoinTask子類后,創建一個代表所有任務的對象,并將之傳遞給一個ForkJoinPool實例的invoke()方法。

由模糊到清晰

為了輔助理解Fork/Join框架是如何工作的,我們使用一個案例來進行說明:比如對一張圖片進行模糊處理。我們用一個整型數組表示圖片,其中的每個數值代表一個像素的顏色。被模糊的圖片也用一個同等長度的數組來表示。

執行模糊是通過對代表圖片的每個像素進行處理實現的。計算每個像素與其周圍像素的均值(紅黃藍三原色的均值),計算生成的結果數組就是模糊后的圖片。由于代表圖像的通常都是一個大數組,整個處理過程需要通常會需要很多時間。可以使用Fork/Join框架利用多處理器系統上的并發處理優勢來進行提速。下面是一個可能的實現:

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
package com.zhyea.robin;
 
import java.util.concurrent.RecursiveAction;
 
public class ForkBlur extends RecursiveAction {
 
  private int[] mSource;
  private int mStart;
  private int mLength;
  private int[] mDestination;
 
  // 處理窗口大小; 需要是一個奇數.
  private int mBlurWidth = 15;
 
  public ForkBlur(int[] src, int start, int length, int[] dst) {
    mSource = src;
    mStart = start;
    mLength = length;
    mDestination = dst;
  }
 
  protected void computeDirectly() {
    int sidePixels = (mBlurWidth - 1) / 2;
    for (int index = mStart; index < mStart + mLength; index++) {
      // 計算平均值.
      float rt = 0, gt = 0, bt = 0;
      for (int mi = -sidePixels; mi <= sidePixels; mi++) {
 
        int mindex = Math.min(Math.max(mi + index, 0), mSource.length - 1);
 
        int pixel = mSource[mindex];
        rt += (float) ((pixel & 0x00ff0000) >> 16) / mBlurWidth;
        gt += (float) ((pixel & 0x0000ff00) >> 8) / mBlurWidth;
        bt += (float) ((pixel & 0x000000ff) >> 0) / mBlurWidth;
      }
 
      // 重組目標像素.
      int dpixel = (0xff000000) |
          (((int) rt) << 16) |
          (((int) gt) << 8) |
          (((int) bt) << 0);
      mDestination[index] = dpixel;
    }
  }
  
  ....
}

現在實現抽象方法compute(),在這個方法中既實現了模糊操作,也實現了將一個任務拆分成兩個小任務。這里僅是簡單依據數組長度來決定是直接執行任務還是將之拆分成兩個小任務:

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
protected static int sThreshold = 100000;
 
protected void compute() {
  if (mLength < sThreshold) {
    computeDirectly();
    return;
  }
 
  int split = mLength / 2;
 
  invokeAll(new ForkBlur(mSource, mStart, split, mDestination),
      new ForkBlur(mSource, mStart + split, mLength - split,
          mDestination));
}

因為上面這些方法的實現是定義在RecursiveAction的一個子類中,可以直接在一個ForkJoinPool中創建并運行任務。具體步驟如下:

1. 創建一個代表要執行的任務的對象:

?
1
2
3
// src 表示源圖片像素的數組
// dst 表示生成的圖片的像素
ForkBlur fb = new ForkBlur(src, 0, src.length, dst);

2. 創建一個運行任務的ForkJoinPool實例:

ForkJoinPool pool = new ForkJoinPool();

3. 運行任務:

pool.invoke(fb);

在源代碼中還包含了一些創建目標圖片的代碼。具體參考ForkBlur示例。

標準實現

要使用Fork/Join框架按自定義的算法在多核系統上執行并發任務當然需要實現自定義的類了(比如之前我們實現的ForkBlur類)。除此之外,在JavaSE中已經在廣泛使用Fork/Join框架的一些特性了。比如Java8中的java.util.Arrays類的parallelSort()方法就使用了Fork/Join框架。具體可以參考Java API文檔。

Fork/Join框架的另一個實現在java.util.streams包下,這也是java8的Lambda特性的一部分。

延伸 · 閱讀

精彩推薦
主站蜘蛛池模板: 人禽l交免费视频 | 综合图区亚洲 | 法国性经典xxxhd | 超碰九色 | 奇米影视亚洲精品一区 | 国产精品久久久久久久久久电影 | 色999久久久精品人人澡69 | 91高清国产 | 高清国产免费 | 热99re久久免费视精品频软件 | 精品一区二区三区免费毛片爱 | 国产免费一级大片 | 日韩视频一区二区在线观看 | 成人在线视频免费观看 | 一级做a爱片久久 | 久久精品艹 | 午夜影视一区二区 | 色污视频 | 国产精品刺激对白麻豆99 | 日本精品中文字幕 | 久久久久久久一区二区 | 色交视频 | 国产成人强伦免费视频网站 | 欧美视频一二三区 | 热99精品视频 | 免费午夜视频 | 成人黄色网址 | 欧美中文日韩 | 国产成人精品一区二区视频免费 | 国产高潮好爽受不了了夜色 | 极品xxxx欧美一区二区 | 精品亚洲在线 | 日本一区二区不卡在线观看 | 久久亚色| 国产美女做爰免费视 | 国产亚洲欧美日韩高清 | 羞羞视频一区二区 | 精品国产一区二区亚洲人成毛片 | 羞羞的视频免费在线观看 | 国产黄色免费网站 | 久久久久久亚洲综合影院红桃 |