19. 線程池

Android 中的線程池就是 java 中的線程池,即 ThreadPoolExecutor 類搀绣。

Java 通過(guò) Executors 提供四種線程池:

  1. newCachedThreadPool

    創(chuàng)建一個(gè)可緩存的線程池

  2. newFixedThreadPool

    創(chuàng)建一個(gè)定長(zhǎng)線程池栋艳,可控制線程最大并發(fā)數(shù)恰聘,超出的線程會(huì)在隊(duì)列中等待。

  3. newScheduledThreadPool

    創(chuàng)建一個(gè)定長(zhǎng)線程池吸占,支持定時(shí)及周期性任務(wù)執(zhí)行晴叨。

  4. newSingleThreadExecutor

    創(chuàng)建一個(gè)單線程化的線程池,它只會(huì)用唯一的工作線程來(lái)執(zhí)行任務(wù)矾屯,保證所有任務(wù)按照指定順序(FIFO, LIFO, 優(yōu)先級(jí))執(zhí)行兼蕊。

Executors 是一個(gè)線程池的工具類,上面四個(gè)方法最終都是調(diào)用的 ThreadPoolExecutor件蚕。理解 ThreadPoolExecutor 就可以完全理解線程池孙技。

線程池解決了 2 個(gè)問(wèn)題:

  1. 當(dāng)執(zhí)行大量異步任務(wù)時(shí),通過(guò)減少每個(gè)任務(wù)的調(diào)用開(kāi)銷來(lái)提高性能排作。
  2. 提供了限制管理線程資源的方法牵啦。

ThreadPoolExecutor

public class ThreadPoolExecutor extends AbstractExecutorService 

構(gòu)造

    /**
     *
     * @param corePoolSize  保持在線程池中的線程數(shù),即使線程處于空閑狀態(tài)妄痪,除非設(shè)置了 allowCoreThreadTimeOut
     * @param maximumPoolSize 線程最大數(shù)量
     * @param keepAliveTime 線程數(shù)大于 corePoolSize 時(shí)哈雏,線程池中的空閑線程等待新任務(wù)執(zhí)行的最大時(shí)間,超過(guò)這個(gè)時(shí)間衫生,空閑線程就會(huì)被終止僧著。
     * @param unit keepAliveTime 的時(shí)間單位
     * @param workQueue 任務(wù)隊(duì)列
     * @param threadFactory 線程工廠
     * @param handler 線程被阻塞時(shí)的處理者,因?yàn)榈竭_(dá)了線程的最大數(shù)量或者任務(wù)隊(duì)列滿了障簿。
     */
    public ThreadPoolExecutor(int corePoolSize,
                              int maximumPoolSize,
                              long keepAliveTime,
                              TimeUnit unit,
                              BlockingQueue<Runnable> workQueue,
                              ThreadFactory threadFactory,
                              RejectedExecutionHandler handler) {
        
        // 檢查參數(shù)
        if (corePoolSize < 0 ||
                maximumPoolSize <= 0 ||
                maximumPoolSize < corePoolSize ||
                keepAliveTime < 0)
            throw new IllegalArgumentException();
        if (workQueue == null || threadFactory == null || handler == null)
            throw new NullPointerException();
        
        // 賦值
        this.corePoolSize = corePoolSize;
        this.maximumPoolSize = maximumPoolSize;
        this.workQueue = workQueue;
        this.keepAliveTime = unit.toNanos(keepAliveTime);
        this.threadFactory = threadFactory;
        this.handler = handler;
    }

執(zhí)行流程

執(zhí)行流程可以用這張圖來(lái)概括:


image

通過(guò) execute()方法執(zhí)行線程池,內(nèi)部會(huì)自動(dòng)根據(jù)我們構(gòu)造參數(shù)的信息分配處理 Runnable栅迄。

    public void execute(Runnable command) {
        // Runnable 不可以為空
        if (command == null)
            throw new NullPointerException();
        
        // ctl 是一個(gè) AtomicInteger站故,用來(lái)記錄正在運(yùn)行的線程數(shù)量
        int c = ctl.get();
        // 當(dāng)前核心的線程數(shù)量<corePoolSize,就新開(kāi)一個(gè)線程將任務(wù)放進(jìn)去
        if (workerCountOf(c) < corePoolSize) {
            if (addWorker(command, true))
                return;
            c = ctl.get();
        }
        
        // 核心線程正在運(yùn)行,且任務(wù)隊(duì)列沒(méi)滿
        if (isRunning(c) && workQueue.offer(command)) {
            // 重新檢查核心線程
            int recheck = ctl.get();
            // 核心線程沒(méi)有運(yùn)行就移除任務(wù)西篓,實(shí)施拒絕策略
            if (! isRunning(recheck) && remove(command))
                reject(command);
            // 核心線程數(shù)量為0愈腾,開(kāi)一個(gè)線程
            else if (workerCountOf(recheck) == 0)
                addWorker(null, false);
        }
        // 任務(wù)隊(duì)列
        else if (!addWorker(command, false))
            reject(command);
    }

參考

深入理解Java之線程池

Android 開(kāi)發(fā)藝術(shù)探索 11.3

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市岂津,隨后出現(xiàn)的幾起案子虱黄,更是在濱河造成了極大的恐慌,老刑警劉巖吮成,帶你破解...
    沈念sama閱讀 206,482評(píng)論 6 481
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件橱乱,死亡現(xiàn)場(chǎng)離奇詭異,居然都是意外死亡粱甫,警方通過(guò)查閱死者的電腦和手機(jī)泳叠,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 88,377評(píng)論 2 382
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái)茶宵,“玉大人危纫,你說(shuō)我怎么就攤上這事∥谑” “怎么了种蝶?”我有些...
    開(kāi)封第一講書(shū)人閱讀 152,762評(píng)論 0 342
  • 文/不壞的土叔 我叫張陵,是天一觀的道長(zhǎng)瞒大。 經(jīng)常有香客問(wèn)我螃征,道長(zhǎng),這世上最難降的妖魔是什么糠赦? 我笑而不...
    開(kāi)封第一講書(shū)人閱讀 55,273評(píng)論 1 279
  • 正文 為了忘掉前任会傲,我火速辦了婚禮,結(jié)果婚禮上拙泽,老公的妹妹穿的比我還像新娘淌山。我一直安慰自己,他們只是感情好顾瞻,可當(dāng)我...
    茶點(diǎn)故事閱讀 64,289評(píng)論 5 373
  • 文/花漫 我一把揭開(kāi)白布泼疑。 她就那樣靜靜地躺著,像睡著了一般荷荤。 火紅的嫁衣襯著肌膚如雪退渗。 梳的紋絲不亂的頭發(fā)上,一...
    開(kāi)封第一講書(shū)人閱讀 49,046評(píng)論 1 285
  • 那天蕴纳,我揣著相機(jī)與錄音会油,去河邊找鬼。 笑死古毛,一個(gè)胖子當(dāng)著我的面吹牛翻翩,可吹牛的內(nèi)容都是我干的都许。 我是一名探鬼主播,決...
    沈念sama閱讀 38,351評(píng)論 3 400
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼嫂冻,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼胶征!你這毒婦竟也來(lái)了?” 一聲冷哼從身側(cè)響起桨仿,我...
    開(kāi)封第一講書(shū)人閱讀 36,988評(píng)論 0 259
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤睛低,失蹤者是張志新(化名)和其女友劉穎,沒(méi)想到半個(gè)月后服傍,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體钱雷,經(jīng)...
    沈念sama閱讀 43,476評(píng)論 1 300
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 35,948評(píng)論 2 324
  • 正文 我和宋清朗相戀三年伴嗡,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了急波。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 38,064評(píng)論 1 333
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡瘪校,死狀恐怖澄暮,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情阱扬,我是刑警寧澤泣懊,帶...
    沈念sama閱讀 33,712評(píng)論 4 323
  • 正文 年R本政府宣布,位于F島的核電站麻惶,受9級(jí)特大地震影響馍刮,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜窃蹋,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 39,261評(píng)論 3 307
  • 文/蒙蒙 一卡啰、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧警没,春花似錦匈辱、人聲如沸。這莊子的主人今日做“春日...
    開(kāi)封第一講書(shū)人閱讀 30,264評(píng)論 0 19
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)。三九已至树酪,卻和暖如春浅碾,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背续语。 一陣腳步聲響...
    開(kāi)封第一講書(shū)人閱讀 31,486評(píng)論 1 262
  • 我被黑心中介騙來(lái)泰國(guó)打工垂谢, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人疮茄。 一個(gè)月前我還...
    沈念sama閱讀 45,511評(píng)論 2 354
  • 正文 我出身青樓埂陆,卻偏偏與公主長(zhǎng)得像苛白,于是被迫代替她去往敵國(guó)和親。 傳聞我的和親對(duì)象是個(gè)殘疾皇子焚虱,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 42,802評(píng)論 2 345

推薦閱讀更多精彩內(nèi)容