Android 線程池的類型、區(qū)別以及為何要用線程池

每個 Android 應(yīng)用進程在創(chuàng)建時赴涵,會同時創(chuàng)建一個線程媒怯,我們稱之為主線程,負責(zé)更新 UI 界面以及和處理用戶之間的交互髓窜,因此扇苞,在 Android 中,我們又稱之為 UI 線程寄纵。一個進程中 UI 線程只有一個鳖敷,為了不造成界面卡頓、提高用戶體驗程拭,我們勢必要將一些耗時操作交由子線程來執(zhí)行定踱。

使用子線程的方式主要分兩種:

  • 直接使用 ThreadRunnable 等創(chuàng)建子并使用線程
  • 使用線程池創(chuàng)建并使用子線程

線程池是什么
線程池是指在初始化一個多線程應(yīng)用程序過程中創(chuàng)建一個線程集合,然后在需要執(zhí)行新的任務(wù)時重用這些線程而不是新創(chuàng)建一個線程恃鞋。線程池中線程的數(shù)量通常完全取決于可用內(nèi)存數(shù)量和應(yīng)用程序的需求崖媚;每個線程都有被分配一個任務(wù),一旦任務(wù)完成了恤浪,線程回到池子中并等待下一次分配任務(wù)畅哑。

一般情況下,推薦使用線程池來創(chuàng)建和使用子線程资锰,不建議使用第一種方式敢课。

為何要用線程池

上面說了,在 Android 開發(fā)過程中绷杜,建議使用線程池來創(chuàng)建和使用子線程直秆,那么使用線程池的好處有哪些呢?

  • 線程池改進了一個應(yīng)用程序的響應(yīng)時間鞭盟。由于線程池中的線程已經(jīng)準(zhǔn)備好且等待被分配任務(wù)圾结,應(yīng)用程序可以直接拿來使用而不用新建一個線程。
  • 線程池節(jié)省了 CLR 為每個短生存周期任務(wù)創(chuàng)建一個完整的線程開銷并可以在任務(wù)完成后回收資源齿诉。
  • 線程池根據(jù)當(dāng)前在系統(tǒng)中運行的進程來優(yōu)化線程片筝野。
  • 線程池允許我們開啟多個任務(wù)而不用為每個線程設(shè)置屬性晌姚。
  • 線程池允許我們?yōu)檎趫?zhí)行的任務(wù)的程序參數(shù)傳遞一個包含狀態(tài)信息的對象引用。
  • 線程池可以用來解決處理一個特定請求最大線程數(shù)量限制問題歇竟。
  • 系統(tǒng)分配給每個應(yīng)用的線程棧是固定的挥唠,使用線程池可以有效地避免線程棧溢出引起的應(yīng)用崩潰。

Android 中的線程池

Android 中線程池的真正實現(xiàn)是 ThreadPoolExecutor焕议,其構(gòu)造方法有 5 個宝磨,通過一系列參數(shù)來配置線程池。下面介紹一個比較常用的構(gòu)造方法及其參數(shù)的含義盅安。

public ThreadPoolExecutor(int corePoolSize,
                          int maximumPoolSize,
                          long keepAliveTime,
                          TimeUnit unit,
                          BlockingQueue<Runnable> workQueue,
                          ThreadFactory threadFactory)
參數(shù) 含義
corePoolSize int: 線程池的核心線程數(shù)唤锉,默認情況下,核心線程回一直在線程池中存活别瞭,即使他們處于閑置狀態(tài)窿祥。如果將 allowCoreThreadTimeOut 的屬性設(shè)置為 true,那么閑置的核心線程在等待新任務(wù)到來時會有超時策略蝙寨,這個時間間隔由 keepAliveTime 所指定晒衩,當(dāng)?shù)却龝r間超過 keepAliveTime 所指定的時長后,核心線程就會被終止籽慢。
maximumPoolSize int: 線程池中允許的線程最大數(shù)量浸遗,當(dāng)活動線程達到這個數(shù)值后,后續(xù)的新任務(wù)會被阻塞箱亿。
keepAliveTime long: 非核心線程閑置時的超時時長,超過這個時長弃秆,非核心線程就會被回收届惋。當(dāng) allowCoreThreadTimeOut 屬性被設(shè)置為 true 時,該參數(shù)同樣會作用于核心線程菠赚。
unit TimeUnit: keepAliveTime 參數(shù)的時間單位 脑豹,參數(shù)為 TimeUnit 的枚舉,常見的有 TimeUnit.MILLISECONDS(毫秒)衡查、TimeUnit.SECOND(秒) 等瘩欺。
workQueue BlockingQueue: 線程池中的任務(wù)隊列,通過線程池的 execute 方法提交的 Runnable 對象會存儲在這個參數(shù)中拌牲。
threadFactory ThreadFactory: 創(chuàng)建線程的線程工廠俱饿。ThreadFactory 是一個接口,只有一個方法:Thread newThread (Runnable r)
Throws
IllegalArgumentException 符合以下任一條件塌忽,則拋出此異常: corePoolSize < 0 keepAliveTime < 0 maximumPoolSize <= 0 maximumPoolSize < corePoolSize
NullPointerException 當(dāng) workQueue 或者 threadFactory 為 null 時拍埠,拋出此異常。

ThreadPoolExecutor 執(zhí)行任務(wù)時大致遵循如下規(guī)則:

  1. 如果線程池中的線程數(shù)量未達到核心線程的數(shù)量土居,那么會直接啟動一個核心線程來執(zhí)行任務(wù)枣购。
  2. 如果線程池中的線程數(shù)量已經(jīng)達到或超過核心線程的數(shù)量嬉探,那么任務(wù)會被插入到任務(wù)隊列中等待執(zhí)行。
  3. 如果步驟 2 中無法將任務(wù)插入到任務(wù)隊列棉圈,則表示任務(wù)隊列已滿涩堤。此時,如果線程數(shù)量未達到 maximumPoolSize 值分瘾,則會立即啟動一個非核心線程來執(zhí)行任務(wù)胎围。
  4. 如果步驟 3 中線程數(shù)量大于或等于 maximumPoolSize 值,則拒絕執(zhí)行次任務(wù)芹敌,此時 ThreadPoolExecutor 會調(diào)用 RejectedExecutionHandlerrejectedExecution 來通知調(diào)用者痊远。

RejectedExecutionHandler 是線程池持有的一個對象,用于不能由 ThreadPoolExecutor 執(zhí)行的任務(wù)的處理

Android 中線程池的類型及區(qū)別

Android 中最常見線程池有四種氏捞,分別是:FixedThreadPool碧聪、CacheThreadPool、ScheduledThreadPool 以及 SingleThreadPool液茎,它們都直接或間接的通過配置 ThreadPoolExecutor 來實現(xiàn)自己的功能特性逞姿。

1. FixedThreadPool

線程數(shù)量固定的線程池,通過 ExecutorsnewFixedThreadPool 方法創(chuàng)建捆等。有兩個重載的創(chuàng)建方法:

public static ExecutorService newFixedThreadPool(int nThreads) {
    return new ThreadPoolExecutor(nThreads, nThreads,
                                  0L, TimeUnit.MILLISECONDS,
                                  new LinkedBlockingQueue<Runnable>());
}
public static ExecutorService newFixedThreadPool(int nThreads, ThreadFactory threadFactory) {
    return new ThreadPoolExecutor(nThreads, nThreads,
                                  0L, TimeUnit.MILLISECONDS,
                                  new LinkedBlockingQueue<Runnable>(),
                                  threadFactory);
}

除非線程池被關(guān)閉滞造,否則線程不會被回收,即時線程處于空閑狀態(tài)栋烤。如果在所有線程都處于活動狀態(tài)時提交額外的任務(wù)谒养,它們將在隊列中等待,直到有一個線程可用為止明郭。由創(chuàng)建方法可知买窟,F(xiàn)ixedThreadPool 只有核心線程并且這個核心線程沒有超時機制(keepAliveTime 參數(shù)為 0L),加上線程不會被回收薯定,因此使用此類線程池可以快速地響應(yīng)外界的請求始绍。

2. CacheThreadPool

線程數(shù)量不定的線程池,通過 ExecutorsnewCachedThreadPool 方法創(chuàng)建话侄。有兩個重載的創(chuàng)建方法:

public static ExecutorService newCachedThreadPool() {
    return new ThreadPoolExecutor(/* corePoolSize*/ 0, Integer.MAX_VALUE,
                                  60L, TimeUnit.SECONDS,
                                  new SynchronousQueue<Runnable>());
}
public static ExecutorService newCachedThreadPool(ThreadFactory threadFactory) {
    return new ThreadPoolExecutor(/* corePoolSize*/ 0, Integer.MAX_VALUE,
                                  60L, TimeUnit.SECONDS,
                                  new SynchronousQueue<Runnable>(),
                                  threadFactory);
}

CacheThreadPool 有以下個特征:

  • 沒有核心線程( corePoolSize 參數(shù)為 0 )亏推,只有非核心線程且非核心線程的數(shù)量為 Integer.MAX_VALUE,這就相當(dāng)于非核心線程的數(shù)量可以無限大年堆。
  • 線程池的線程處于空閑狀態(tài)時吞杭,線程池會重用空閑的線程來處理新任務(wù),否則創(chuàng)建新的線程來處理嘀韧,新創(chuàng)建的線程會添加到線程池中篇亭。這將提高執(zhí)行許多短期異步任務(wù)的程序性能。
  • 閑置時間超過 60 秒的空閑線程會被回收(keepAliveTime 參數(shù)為 60L )锄贷。因此译蒂,閑置時間足夠長的 CacheThreadPool 也不會消耗任何系統(tǒng)資源曼月。

3. ScheduledThreadPool

核心線程數(shù)量固定,非核心線程數(shù)量不定的線程池柔昼,通過 Executorsnewscheduledthreadpool 方法創(chuàng)建哑芹。有兩個重載的創(chuàng)建方法:

public static ScheduledExecutorService newScheduledThreadPool(int corePoolSize) {
    return new ScheduledThreadPoolExecutor(corePoolSize);
}
public static ScheduledExecutorService newScheduledThreadPool(
        int corePoolSize, ThreadFactory threadFactory) {
    return new ScheduledThreadPoolExecutor(corePoolSize, threadFactory);
}

ScheduledThreadPool 相比較其他三種線程池,有特殊性捕透,由 ScheduledThreadPoolExecutor 實現(xiàn)聪姿, newscheduledthreadpool 方法也是通過創(chuàng)建 ScheduledThreadPoolExecutor 的實例來完成線程池的創(chuàng)建,代碼如下:

public ScheduledThreadPoolExecutor(int corePoolSize) {
    super(corePoolSize, Integer.MAX_VALUE,
          DEFAULT_KEEPALIVE_MILLIS, MILLISECONDS,
          new DelayedWorkQueue());
}
public ScheduledThreadPoolExecutor(int corePoolSize, ThreadFactory threadFactory) {    super(corePoolSize, Integer.MAX_VALUE,
          DEFAULT_KEEPALIVE_MILLIS, MILLISECONDS,
          new DelayedWorkQueue(), threadFactory);
}

ScheduledThreadPoolExecutor 的構(gòu)造函數(shù)可知乙嘀, ScheduledThreadPool 的核心線程數(shù)量是固定的末购,由傳入的 corePoolSize 參數(shù)決定,非核心線程數(shù)量可以無限大虎谢。非核心線程閑置回收的超時時間為 10秒DEFAULT_KEEPALIVE_MILLIS 的值為 10L )盟榴。這類線程主要用于執(zhí)行定時任務(wù)或者具有周期性的重復(fù)任務(wù)。

4. SingleThreadPool

只有一個核心線程婴噩,通過 Executorsnewsinglethreadexecutor 方法創(chuàng)建擎场。有兩個重載的創(chuàng)建方法:

public static ExecutorService newSingleThreadExecutor() {
    return new FinalizableDelegatedExecutorService
        (new ThreadPoolExecutor(/* corePoolSize*/ 1, /* maximumPoolSize*/ 1,
                                /* keepAliveTime*/ 0L, TimeUnit.MILLISECONDS,
                                new LinkedBlockingQueue<Runnable>()));
}
public static ExecutorService newSingleThreadExecutor(ThreadFactory threadFactory) {
    return new FinalizableDelegatedExecutorService
        (new ThreadPoolExecutor(/* corePoolSize*/ 1, /* maximumPoolSize*/ 1,
                                /* keepAliveTime*/ 0L, TimeUnit.MILLISECONDS,
                                new LinkedBlockingQueue<Runnable>(),
                                threadFactory));
}

SingleThreadPool 由 代理類 FinalizableDelegatedExecutorService 創(chuàng)建。該線程池只有一個線程(核心線程)几莽,并且該線程池的任務(wù)隊列是無上限的迅办,這就確保了所有的任務(wù)都在同一個線程中順序執(zhí)行。

注意章蚣,如果由于在執(zhí)行期間出現(xiàn)故障而導(dǎo)致該線程終止站欺,那么如果需要執(zhí)行后續(xù)任務(wù),則新線程將取而代之纤垂。

四類線程池的區(qū)別

上面分別對 Android 中常見的 4 種線程池進行了簡單的介紹镊绪,除了這 4 種系統(tǒng)提供的線程池外,我們在使用的過程中洒忧,也可以根據(jù)需要直接通過 ThreadPoolExecutor 的構(gòu)造函數(shù)來靈活的配置線程池。那么够颠,上述的 4 種線程池熙侍,其區(qū)別在哪呢?了解其區(qū)別有助于我們?nèi)ミx擇更為合適的線程池或者直接通過 ThreadPoolExecutor 來配置更靈活的線程池履磨。

FixedThreadPool 線程固定蛉抓,且不會被回收,能夠更快的響應(yīng)外界請求剃诅。

CachedThreadPool 只有非核心線程巷送,且線程數(shù)相當(dāng)于無限大,任何任務(wù)都會被立即執(zhí)行矛辕。比較適合執(zhí)行大量的耗時較少的任務(wù)笑跛。

ScheduledThreadPool 主要用于執(zhí)行定時任務(wù)或者具有周期性的重復(fù)任務(wù)付魔。

SingleThreadPool 只有一個核心線程,確保所有任務(wù)都在同一線程中按順序完成飞蹂。因此不需要處理線程同步的問題几苍。

參考

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末陈哑,一起剝皮案震驚了整個濱河市妻坝,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌惊窖,老刑警劉巖刽宪,帶你破解...
    沈念sama閱讀 216,544評論 6 501
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異界酒,居然都是意外死亡圣拄,警方通過查閱死者的電腦和手機,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,430評論 3 392
  • 文/潘曉璐 我一進店門盾计,熙熙樓的掌柜王于貴愁眉苦臉地迎上來售担,“玉大人,你說我怎么就攤上這事署辉∽迕” “怎么了?”我有些...
    開封第一講書人閱讀 162,764評論 0 353
  • 文/不壞的土叔 我叫張陵哭尝,是天一觀的道長哥攘。 經(jīng)常有香客問我,道長材鹦,這世上最難降的妖魔是什么逝淹? 我笑而不...
    開封第一講書人閱讀 58,193評論 1 292
  • 正文 為了忘掉前任,我火速辦了婚禮桶唐,結(jié)果婚禮上栅葡,老公的妹妹穿的比我還像新娘。我一直安慰自己尤泽,他們只是感情好欣簇,可當(dāng)我...
    茶點故事閱讀 67,216評論 6 388
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著坯约,像睡著了一般熊咽。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上闹丐,一...
    開封第一講書人閱讀 51,182評論 1 299
  • 那天横殴,我揣著相機與錄音,去河邊找鬼卿拴。 笑死衫仑,一個胖子當(dāng)著我的面吹牛梨与,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播惑畴,決...
    沈念sama閱讀 40,063評論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼蛋欣,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了如贷?” 一聲冷哼從身側(cè)響起陷虎,我...
    開封第一講書人閱讀 38,917評論 0 274
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎杠袱,沒想到半個月后尚猿,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 45,329評論 1 310
  • 正文 獨居荒郊野嶺守林人離奇死亡楣富,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 37,543評論 2 332
  • 正文 我和宋清朗相戀三年凿掂,在試婚紗的時候發(fā)現(xiàn)自己被綠了。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片纹蝴。...
    茶點故事閱讀 39,722評論 1 348
  • 序言:一個原本活蹦亂跳的男人離奇死亡庄萎,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出塘安,到底是詐尸還是另有隱情糠涛,我是刑警寧澤,帶...
    沈念sama閱讀 35,425評論 5 343
  • 正文 年R本政府宣布兼犯,位于F島的核電站忍捡,受9級特大地震影響,放射性物質(zhì)發(fā)生泄漏切黔。R本人自食惡果不足惜砸脊,卻給世界環(huán)境...
    茶點故事閱讀 41,019評論 3 326
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望纬霞。 院中可真熱鬧凌埂,春花似錦、人聲如沸诗芜。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,671評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽绢陌。三九已至,卻和暖如春熔恢,著一層夾襖步出監(jiān)牢的瞬間脐湾,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 32,825評論 1 269
  • 我被黑心中介騙來泰國打工叙淌, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留秤掌,地道東北人愁铺。 一個月前我還...
    沈念sama閱讀 47,729評論 2 368
  • 正文 我出身青樓,卻偏偏與公主長得像闻鉴,于是被迫代替她去往敵國和親茵乱。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點故事閱讀 44,614評論 2 353

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

  • 前段時間遇到這樣一個問題孟岛,有人問微信朋友圈的上傳圖片的功能怎么做才能讓用戶的等待時間較短瓶竭,比如說一下上傳9張圖片,...
    加油碼農(nóng)閱讀 1,201評論 0 2
  • Android 自定義View的各種姿勢1 Activity的顯示之ViewRootImpl詳解 Activity...
    passiontim閱讀 172,077評論 25 707
  • Android中的線程 線程,在Android中是非常重要的渠羞,主線程處理UI界面斤贰,子線程處理耗時操作。如果在主線程...
    shenhuniurou閱讀 755評論 0 3
  • 寫給小紅:沒有見過你之前次询,蘆葦姐姐在群里發(fā)你光頭的那個表情包荧恍,我一直覺得是網(wǎng)上的表情包!結(jié)果你差不多....就是那...
    外聯(lián)一家人閱讀 152評論 0 1
  • -2- “喵喵咖啡館”葉筱筱和閨蜜約好在這里復(fù)習(xí)屯吊,迎接高考送巡。閨蜜意料之中的遲到了,葉筱筱已經(jīng)習(xí)以為常了盒卸。 ...
    涔曦閱讀 769評論 2 2