Java線程池:ThreadPoolExecutor詳解

用過java線程的同學都應該大致了解,在java中,為了合理使用線程以合理利用資源、提高吞吐率以及加快響應時間,通常會使用java線程池,因為線程池架構設計合理,比起自己創建線程可能花銷巨大來講,線程池是一個很好的選擇。

作為一只喜歡研究源碼的程序猿,就我所學,來講講java線程池是如何鞏工作的。

一.4種線程池

首先,java線程池為我們量身定制了4中拿來即用的線程池:

1.newSingleThreadExecutor:

public static ExecutorServicenewSingleThreadExecutor() {

????????????return new FinalizableDelegatedExecutorService

? ? ? ?????????????(new ThreadPoolExecutor(1, 1,

? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? 0L, TimeUnit.MILLISECONDS,

? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? new LinkedBlockingQueue()));

}


2.newFixedThreadPool:

public static ExecutorServicenewFixedThreadPool(int nThreads) {

????????????return new ThreadPoolExecutor(nThreads, nThreads,

? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? 0L, TimeUnit.MILLISECONDS,

? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? new LinkedBlockingQueue());

}

3.newCachedThreadPool:

public static ExecutorServicenewCachedThreadPool() {

????????????return new ThreadPoolExecutor(0, Integer.MAX_VALUE,

? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? 60L, TimeUnit.SECONDS,

? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? new SynchronousQueue());

}

4. newScheduledThreadPool:

public static ScheduledExecutorServicenewSingleThreadScheduledExecutor() {

????????????return new DelegatedScheduledExecutorService

????????????????????????(new ScheduledThreadPoolExecutor(1));

}

可以發現:除了newScheduledThreadPool,其他三個都是ThreadPoolExecutor的一種特殊實現。

二.ThreadPoolExecutor

public ThreadPoolExecutor(int corePoolSize,

? ? ? ? ? ? ? ? ? ? ? ? ? int maximumPoolSize,

? ? ? ? ? ? ? ? ? ? ? ? ? long keepAliveTime,

? ? ? ? ? ? ? ? ? ? ? ? ? TimeUnit unit,

? ? ? ? ? ? ? ? ? ? ? ? ? BlockingQueue workQueue,

? ? ? ? ? ? ? ? ? ? ? ? ? ThreadFactory threadFactory,

? ? ? ? ? ? ? ? ? ? ? ? ? RejectedExecutionHandler handler) {

if (corePoolSize <0 ||

maximumPoolSize <=0 ||

maximumPoolSize < corePoolSize ||

keepAliveTime <0)

throw new IllegalArgumentException();

? ? if (workQueue ==null || threadFactory ==null || handler ==null)

throw new NullPointerException();

? ? this.acc = System.getSecurityManager() ==null ?

null :

AccessController.getContext();

? ? this.corePoolSize = corePoolSize;

? ? this.maximumPoolSize = maximumPoolSize;

? ? this.workQueue = workQueue;

? ? this.keepAliveTime = unit.toNanos(keepAliveTime);

? ? this.threadFactory = threadFactory;

? ? this.handler = handler;

}

1.構造函數參數說明:

1).corePoolSize:線程池中的核心線程數

2).maximumPoolSize:線程池中的最大線程數

3).keepAliveTime: 線程池中的線程的存活時間

4).unit: keepAliveTime的時間單位

5).workQueue: BlockingQueue的實現,用于存儲任務

6).threadFactory:自定義線程的創建工廠

7).handler:線程池的飽和策略,當阻塞隊列滿了,且沒有空閑的線程,繼續提交任務時,必須進行處理,默認的方式是拋出RejectedExecutionHandler異常

2.內部狀態變量說明:

jdk1.8就是用一個int型來表示線程池的運行狀態和運行任務數量,一個int一共32位,前3位表示運行狀態,后29位表示運行任務線程數;運行狀態和運行任務線程數量分別通過runStateOf(int c)和workerCountOf(int c)來獲取,都是通過&操作來計算的。各個值的初始值分為如下:

private final AtomicInteger ctl = new AtomicInteger(ctlOf(RUNNING, 0));

private static final int COUNT_BITS = Integer.SIZE - 3;

private static final int CAPACITY = (1 << COUNT_BITS) - 1;

// runState is stored in the high-order bits

private static final int RUNNING = -1 << COUNT_BITS;

private static final int SHUTDOWN = 0 << COUNT_BITS;

private static final int STOP = 1 << COUNT_BITS;

private static final int TIDYING = 2 << COUNT_BITS;

private static final int TERMINATED = 3 << COUNT_BITS;

// Packing and unpacking ctl

private static int runStateOf(int c) { return c & ~CAPACITY; }

private static int workerCountOf(int c) { return c & CAPACITY; }

private static int ctlOf(int rs, int wc) { return rs | wc; }

?1).ctl:包裝任務狀態和線程數(高3位表示運行狀態,后29位表示運行任務線程數),初始值:11100000000000000000000000000000,沒多一線程就加1,為原子操作;默認AtomicInteger,如果數量不夠的話,可以自行改成AtomicLong

? 2).COUNT_BITS:29

? 3).CAPACITY:運行任務線程數,值為:000 11111111111111111111111111111

??4).RUNNING:運行狀態,? ? ? ?值為:111 00000000000000000000000000000

??5).SHUTDOWN:關閉狀態,? 值為:0

? 6).STOP: 停止狀態,? ? ? ? ? ? ?值為:001 00000000000000000000000000000

? 7).TIDYING:整理狀態,? ? ? ? ?值為:010 00000000000000000000000000000

? 8).TERMINATED:終止狀態,值為:011 00000000000000000000000000000

3.內部公用函數說明:

1).runStateOf(int c):獲取運行狀態,

2).workerCountOf(int c):獲取運行任務線程數

3).ctl(int rs,int wc): 包裝運行狀態和任務線程數

4).runStateLessThan(int c,int s):是否小于某個狀態

5).runStateAtLesat(int c,int s):是否大于或者等于某個狀態

這些公共函數接下來都會用到,寫在這里是便于理解。

4.提交任務函數:execute(Runnable command)

public void execute(Runnable command){

????if (command == null)

????????throw new NullPointerException();

????int c = ctl.get();

????if (workerCountOf(c) < corePoolSize) {

????????if (addWorker(command, true))

????????????return;

????????c = ctl.get();

????}

????if (isRunning(c) && workQueue.offer(command)) {

????????int recheck = ctl.get();

????????if (! isRunning(recheck) && remove(command))

????????????reject(command);

????????else if (workerCountOf(recheck) == 0)

????????????addWorker(null, false);

?????} else if (!addWorker(command, false))

????????????reject(command);

}

1).如果當前運行任務線程數小于corePoolSize,嘗試另起線程去運行任務。調用addWorker函數會自動檢查運行狀態和運行任務線程數,防止添加線程時出現多線程操作錯誤。

2).如果添加新的線程失敗,那么就將該任務添加到隊列中,同時,還需要檢查運行狀態和運行任務線程數;再次檢查狀態,防止入列期間出現狀態改變情況,如果線程池處于非運行狀態,移除任務;如果沒有運行任務線程數量為0,則起一個新的線程。

3).如果任務不能進入隊列(例如隊列滿了),再次嘗試另起一個線程運行;如果失敗了,使用拒絕策略。

5.addWorker

在execute函數中,addWorker是另起線程去執行任務,它的具體實現如下:

private boolean addWorker(Runnable firstTask, boolean core) {

????retry:

????for (;;) {

????????int c = ctl.get();

????????int rs = runStateOf(c);

????????// Check if queue empty only if necessary.

????????if (rs >= SHUTDOWN &&

????????????! (rs == SHUTDOWN && firstTask == null && ! workQueue.isEmpty()))

????????????????return false;

????????for (;;) {

????????????int wc = workerCountOf(c);

????????????if (wc >= CAPACITY ||?

? ? ? ? ? ? ? ? ? ? wc >= (core ? corePoolSize : maximumPoolSize))

????????????????return false;

????????????if (compareAndIncrementWorkerCount(c))

????????????????break retry;

????????????????c = ctl.get();// Re-read ctl

????????????if (runStateOf(c) != rs)

????????????????continue retry; // else CAS failed due to workerCount change; retry inner loop? ? ? ? ? ?}

????}

????boolean workerStarted = false;

? ? boolean workerAdded = false;

????Worker w = null;

????try {

????????w = new Worker(firstTask);

????????final Thread t = w.thread;

????????if (t != null) {

????????????final ReentrantLock mainLock = this.mainLock;

????????????mainLock.lock();

????????????try { // Recheck while holding lock. // Back out on ThreadFactory failure or if // shut down before lock acquired.

????????????????????int rs = runStateOf(ctl.get());

????????????????????if (rs < SHUTDOWN

????????????????????????????|| (rs == SHUTDOWN && firstTask == null)) {

????????????????????????????if (t.isAlive()) // precheck that t is startable

????????????????????????????????throw new IllegalThreadStateException();

????????????????????????????workers.add(w);

????????????????????????????int s = workers.size();

????????????????????????????if (s > largestPoolSize)

????????????????????????????largestPoolSize = s;

????????????????????????????workerAdded = true;

????????????????????????}

????????????????} finally {

????????????????????mainLock.unlock();

????????????????}

????????????????if (workerAdded) {

????????????????????t.start();

????????????????????workerStarted = true;

????????????????}

????????????}

????????} finally {

????????????if (! workerStarted)

????????????addWorkerFailed(w);

????????????}

????????????return workerStarted;

}

最后編輯于
?著作權歸作者所有,轉載或內容合作請聯系作者
平臺聲明:文章內容(如有圖片或視頻亦包括在內)由作者上傳并發布,文章內容僅代表作者本人觀點,簡書系信息發布平臺,僅提供信息存儲服務。
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市,隨后出現的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 229,117評論 6 537
  • 序言:濱河連續發生了三起死亡事件,死亡現場離奇詭異,居然都是意外死亡,警方通過查閱死者的電腦和手機,發現死者居然都...
    沈念sama閱讀 98,860評論 3 423
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人,你說我怎么就攤上這事。” “怎么了?”我有些...
    開封第一講書人閱讀 177,128評論 0 381
  • 文/不壞的土叔 我叫張陵,是天一觀的道長。 經常有香客問我,道長,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 63,291評論 1 315
  • 正文 為了忘掉前任,我火速辦了婚禮,結果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當我...
    茶點故事閱讀 72,025評論 6 410
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發上,一...
    開封第一講書人閱讀 55,421評論 1 324
  • 那天,我揣著相機與錄音,去河邊找鬼。 笑死,一個胖子當著我的面吹牛,可吹牛的內容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 43,477評論 3 444
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了?” 一聲冷哼從身側響起,我...
    開封第一講書人閱讀 42,642評論 0 289
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后,有當地人在樹林里發現了一具尸體,經...
    沈念sama閱讀 49,177評論 1 335
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內容為張勛視角 年9月15日...
    茶點故事閱讀 40,970評論 3 356
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發現自己被綠了。 大學時的朋友給我發了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 43,157評論 1 371
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 38,717評論 5 362
  • 正文 年R本政府宣布,位于F島的核電站,受9級特大地震影響,放射性物質發生泄漏。R本人自食惡果不足惜,卻給世界環境...
    茶點故事閱讀 44,410評論 3 347
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 34,821評論 0 28
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至,卻和暖如春,著一層夾襖步出監牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 36,053評論 1 289
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個月前我還...
    沈念sama閱讀 51,896評論 3 395
  • 正文 我出身青樓,卻偏偏與公主長得像,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 48,157評論 2 375

推薦閱讀更多精彩內容