android-lite-go 源碼設(shè)計分析

LiteGo:阿里工程師馬天宇開源的迷你的Android異步并發(fā)類庫。

LiteGo 特性

可定義核心并發(fā)線程數(shù)郎嫁,即同一時間并發(fā)的請求數(shù)量秉继。

可定義等待排隊線程數(shù),即超出核心并發(fā)數(shù)后可排隊請求數(shù)量泽铛。

可定義等待隊列進入執(zhí)行狀態(tài)的策略:先來先執(zhí)行尚辑,后來先執(zhí)行。

可定義等待隊列滿載后處理新請求的策略:

  • 拋棄隊列中最新的任務
  • 拋棄隊列中最舊的任務
  • 拋棄當前新任務
  • 直接執(zhí)行(阻塞當前線程)
  • 拋出異常(中斷當前線程)

如何合理的估計同時并發(fā)的CPU數(shù)量盔腔?

參考如何合理地估算線程池大懈懿纭月褥?

最大并發(fā)數(shù)與CPU核心關(guān)系

首先我們經(jīng)常在電腦介紹中聽到雙核四線程,因為支持超線程技術(shù)的CPU瓢喉,單個核心可以同時并發(fā)兩個線程進行協(xié)同工作宁赤,相比單線程效率更高,但也達不到想象中性能翻倍的效果栓票。而不支持超線程技術(shù)的CPU决左,一個核心就只能以一個線程進行運算,手機上沒有超線程技術(shù)所以手機上并發(fā)的最大線程數(shù)就等于核心數(shù)逗载。

如果是CPU密集型應用哆窿,則線程池大小設(shè)置為N+1
如果是IO密集型應用,則線程池大小設(shè)置為2N+1
如果一臺服務器上只部署這一個應用并且只有這一個線程池厉斟,那么這種估算或許合理挚躯,具體還需自行測試驗證。

這里我認同lite go的理念

  • 清閑時線程不要多持擦秽,最好不要超過CPU數(shù)量码荔,根據(jù)具體應用類型和場- 景來決策。
  • 瞬間并發(fā)不要過多感挥,最好保持在CPU數(shù)量左右缩搅,或者可以多幾個問題并不大。
    同時并發(fā)的線程數(shù)量不要過多触幼,最好保持在CPU核數(shù)左右硼瓣,過多了CPU時間片過多的輪轉(zhuǎn)分配造成吞吐量降低,過少了不能充分利用CPU置谦,并發(fā)數(shù)可以適當比CPU核數(shù)多一點沒問題堂鲤。

主要的類

  1. SmartExecutor 類實現(xiàn)了Executor 接口:智能并發(fā)調(diào)度執(zhí)行器

PriorityRunnable 抽象類 子類需要實現(xiàn) Runnable 接口:顧名思義 優(yōu)先級Runnable ;含有int類型成員變量priority媒峡。

  1. SchedulePolicy 枚舉類型:線程池等待隊列進入執(zhí)行狀態(tài)的調(diào)度策略瘟栖,目前有兩種策略,一個是先進先出(隊列)谅阿,另一種是先進后出(棧)
  2. OverloadPolicy 枚舉類型:線程池等待隊列滿載后處理新請求的裝載策略半哟,目前有五種策略
  • 拋棄隊列中最新的任務
  • 拋棄隊列中最舊的任務
  • 拋棄當前新任務
  • 直接執(zhí)行(阻塞當前線程)
  • 拋出異常(中斷當前線程)

在上面所有的類當中SmartExecutor 是這個庫的核心類,接下來主要分析這個類的實現(xiàn)签餐。在SmartExecutor類當中有一個非常重要的屬性ThreadPoolExecutor(線程池執(zhí)行器)這里是代理設(shè)計模式的體現(xiàn)寓涨,先了解一下ThreadPoolExecutor的構(gòu)造函數(shù)參數(shù)含義publicThreadPoolExecutor(int corePoolSize, int maximumPoolSize, long keepAliveTime, TimeUnit unit, BlockingQueue<Runnable> workQueue, ThreadFactory threadFactory, RejectedExecutionHandler handler)
corePoolSize:核心線程數(shù)
核心線程會一直存活,及時沒有任務需要執(zhí)行
當線程數(shù)小于核心線程數(shù)時氯檐,即使有線程空閑戒良,線程池也會優(yōu)先創(chuàng)建新線程處理
設(shè)置allowCoreThreadTimeout=true(默認false)時,核心線程會超時關(guān)閉
maximumPoolSize:最大線程數(shù)
當線程數(shù)>=corePoolSize男摧,且任務隊列已滿時蔬墩。線程池會創(chuàng)建新線程來處理任務
當線程數(shù)=maximumPoolSize译打,且任務隊列已滿時,線程池會拒絕處理任務而拋出異常
keepAliveTime:線程空閑時間
當線程空閑時間達到keepAliveTime時拇颅,線程會退出奏司,直到線程數(shù)量=corePoolSize
如果allowCoreThreadTimeout=true,則會直到線程數(shù)量=0

rejectedExecutionHandler:任務拒絕處理器
兩種情況會拒絕處理任務:
當線程數(shù)已經(jīng)達到maximumPoolSize樟插,切隊列已滿韵洋,會拒絕新任務
當線程池被調(diào)用shutdown()后,會等待線程池里的任務執(zhí)行完畢黄锤,再shutdown搪缨。如果在調(diào)用shutdown()和線程池真正shutdown之間提交任務,會拒絕新任務
線程池會調(diào)用rejectedExecutionHandler來處理這個任務鸵熟。如果沒有設(shè)置默認是AbortPolicy副编,會拋出異常
ThreadPoolExecutor類有幾個內(nèi)部實現(xiàn)類來處理這類情況:
AbortPolicy 丟棄任務,拋運行時異常
CallerRunsPolicy 執(zhí)行任務
DiscardPolicy 忽視流强,什么都不會發(fā)生
DiscardOldestPolicy 從隊列中踢出最先進入隊列(最后一個執(zhí)行)的任務
實現(xiàn)RejectedExecutionHandler接口痹届,可自定義處理器

ThreadPoolExecutor執(zhí)行順序

線程池按以下行為執(zhí)行任務
  1. 當線程數(shù)小于核心線程數(shù)時,創(chuàng)建線程打月。
  2. 當線程數(shù)大于等于核心線程數(shù)队腐,且任務隊列未滿時,將任務放入任務隊列奏篙。
  3. 當線程數(shù)大于等于核心線程數(shù)柴淘,且任務隊列已滿
  4. 若線程數(shù)小于最大線程數(shù),創(chuàng)建線程
  5. 若線程數(shù)等于最大線程數(shù)秘通,拋出異常为严,拒絕任務

android-lite-go
SmartExecutor 的幾個主要方法:

public Future<?> submit(Runnable task)

public <T> Future<T> submit(Runnable task, T result)

public <T> Future<T> submit(Callable<T> task)

public <T> void submit(RunnableFuture<T> task)

public void execute(final Runnable command)
最主要的是 execute 方法,其他幾個方法是將任務封裝為 FutureTask 投入到 execute 方法執(zhí)行充易。因為 FutureTask 本質(zhì)就是一個 RunnableFuture 對象梗脾,兼具 Runnable 和 Future 的特性和功能荸型。
execute 整個過程簡單概括為:

把任務封裝為一個類似“鏈表”的結(jié)構(gòu)體盹靴,執(zhí)行完一個,調(diào)度下一個瑞妇。
加鎖防止并發(fā)時搶奪資源稿静,判斷當前運行任務數(shù)量。
當前任務數(shù)少于并發(fā)最大數(shù)量則投入運行辕狰,若滿載則投入等待隊列尾部改备。
若等待隊列未滿新任務進入排隊,若滿則執(zhí)行滿載處理策略蔓倍。
當一個任務執(zhí)行完悬钳,其尾部通過“鏈接”的方式調(diào)度一個新任務執(zhí)行盐捷。若沒有任務,則結(jié)束默勾。
其中「加鎖」和將任務包裝成「鏈表」是重點碉渡。

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市母剥,隨后出現(xiàn)的幾起案子滞诺,更是在濱河造成了極大的恐慌,老刑警劉巖环疼,帶你破解...
    沈念sama閱讀 212,884評論 6 492
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件习霹,死亡現(xiàn)場離奇詭異,居然都是意外死亡炫隶,警方通過查閱死者的電腦和手機淋叶,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 90,755評論 3 385
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來伪阶,“玉大人爸吮,你說我怎么就攤上這事⊥牛” “怎么了形娇?”我有些...
    開封第一講書人閱讀 158,369評論 0 348
  • 文/不壞的土叔 我叫張陵,是天一觀的道長筹误。 經(jīng)常有香客問我桐早,道長,這世上最難降的妖魔是什么厨剪? 我笑而不...
    開封第一講書人閱讀 56,799評論 1 285
  • 正文 為了忘掉前任哄酝,我火速辦了婚禮,結(jié)果婚禮上祷膳,老公的妹妹穿的比我還像新娘陶衅。我一直安慰自己,他們只是感情好直晨,可當我...
    茶點故事閱讀 65,910評論 6 386
  • 文/花漫 我一把揭開白布搀军。 她就那樣靜靜地躺著,像睡著了一般勇皇。 火紅的嫁衣襯著肌膚如雪罩句。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 50,096評論 1 291
  • 那天敛摘,我揣著相機與錄音门烂,去河邊找鬼。 笑死,一個胖子當著我的面吹牛屯远,可吹牛的內(nèi)容都是我干的蔓姚。 我是一名探鬼主播,決...
    沈念sama閱讀 39,159評論 3 411
  • 文/蒼蘭香墨 我猛地睜開眼慨丐,長吁一口氣:“原來是場噩夢啊……” “哼赂乐!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起咖气,我...
    開封第一講書人閱讀 37,917評論 0 268
  • 序言:老撾萬榮一對情侶失蹤挨措,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后崩溪,有當?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體浅役,經(jīng)...
    沈念sama閱讀 44,360評論 1 303
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 36,673評論 2 327
  • 正文 我和宋清朗相戀三年伶唯,在試婚紗的時候發(fā)現(xiàn)自己被綠了觉既。 大學時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 38,814評論 1 341
  • 序言:一個原本活蹦亂跳的男人離奇死亡乳幸,死狀恐怖瞪讼,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情粹断,我是刑警寧澤符欠,帶...
    沈念sama閱讀 34,509評論 4 334
  • 正文 年R本政府宣布,位于F島的核電站瓶埋,受9級特大地震影響希柿,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜养筒,卻給世界環(huán)境...
    茶點故事閱讀 40,156評論 3 317
  • 文/蒙蒙 一曾撤、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧晕粪,春花似錦挤悉、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,882評論 0 21
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至剩膘,卻和暖如春衅斩,著一層夾襖步出監(jiān)牢的瞬間盆顾,已是汗流浹背怠褐。 一陣腳步聲響...
    開封第一講書人閱讀 32,123評論 1 267
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留您宪,地道東北人奈懒。 一個月前我還...
    沈念sama閱讀 46,641評論 2 362
  • 正文 我出身青樓奠涌,卻偏偏與公主長得像,于是被迫代替她去往敵國和親磷杏。 傳聞我的和親對象是個殘疾皇子溜畅,可洞房花燭夜當晚...
    茶點故事閱讀 43,728評論 2 351

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