java?Executors常用線程池
newFixedThreadPool 固定數(shù)目線程的線程池
public static ExecutorService newFixedThreadPool(int nThreads) { return new ThreadPoolExecutor(nThreads, nThreads, 0L, TimeUnit.MILLISECONDS, new LinkedBlockingQueue<Runnable>()); }
線程池特點(diǎn)
核心線程數(shù)和最大線程數(shù)大小一樣
沒有所謂的非空閑時(shí)間,即keepAliveTime為0
阻塞隊(duì)列為無界隊(duì)列LinkedBlockingQueue
工作機(jī)制
- 提交任務(wù)
- 如果線程數(shù)少于核心線程,創(chuàng)建核心線程執(zhí)行任務(wù)
- 如果線程數(shù)等于核心線程,把任務(wù)添加到LinkedBlockingQueue阻塞隊(duì)列
- 如果線程執(zhí)行完任務(wù),去阻塞隊(duì)列取任務(wù),繼續(xù)執(zhí)行。
使用無界隊(duì)列的線程池會(huì)導(dǎo)致內(nèi)存飆升嗎?
會(huì)的,newFixedThreadPool使用了無界的阻塞隊(duì)列LinkedBlockingQueue,如果線程獲取一個(gè)任務(wù)后,任務(wù)的執(zhí)行時(shí)間比較長,會(huì)導(dǎo)致隊(duì)列的任務(wù)越積越多,導(dǎo)致機(jī)器內(nèi)存使用不停飆升,最終導(dǎo)致OOM。
使用場景
FixedThreadPool 適用于處理CPU密集型的任務(wù),確保CPU在長期被工作線程使用的情況下,盡可能的少的分配線程,即適用執(zhí)行長期的任務(wù)。
newCachedThreadPool 可緩存線程的線程池
public static ExecutorService newCachedThreadPool(ThreadFactory threadFactory) { return new ThreadPoolExecutor(0, Integer.MAX_VALUE, 60L, TimeUnit.SECONDS, new SynchronousQueue<Runnable>(), threadFactory); }
線程池特點(diǎn)
核心線程數(shù)為0
最大線程數(shù)為Integer.MAX_VALUE
阻塞隊(duì)列是SynchronousQueue
非核心線程空閑存活時(shí)間為60秒
當(dāng)提交任務(wù)的速度大于處理任務(wù)的速度時(shí),每次提交一個(gè)任務(wù),就必然會(huì)創(chuàng)建一個(gè)線程。極端情況下會(huì)創(chuàng)建過多的線程,耗盡CPU和內(nèi)存資源。由于空閑 60 秒的線程會(huì)被終止,長時(shí)間保持空閑的 CachedThreadPool 不會(huì)占用任何資源。
工作機(jī)制
- 提交任務(wù)
- 因?yàn)闆]有核心線程,所以任務(wù)直接加到SynchronousQueue隊(duì)列。
- 判斷是否有空閑線程,如果有,就去取出任務(wù)執(zhí)行。
- 如果沒有空閑線程,就新建一個(gè)線程執(zhí)行。
- 執(zhí)行完任務(wù)的線程,還可以存活60秒,如果在這期間,接到任務(wù),可以繼續(xù)活下去;否則,被銷毀。
使用場景
用于并發(fā)執(zhí)行大量短期的小任務(wù)。
newSingleThreadExecutor 單線程的線程池
public static ExecutorService newSingleThreadExecutor() { return new FinalizableDelegatedExecutorService (new ThreadPoolExecutor(1, 1, 0L, TimeUnit.MILLISECONDS, new LinkedBlockingQueue<Runnable>())); }
線程池特點(diǎn)
核心線程數(shù)為1
最大線程數(shù)也為1
阻塞隊(duì)列是LinkedBlockingQueue
keepAliveTime為0
工作機(jī)制
- 提交任務(wù)
- 線程池是否有一條線程在,如果沒有,新建線程執(zhí)行任務(wù)
- 如果有,將任務(wù)加到阻塞隊(duì)列
- 當(dāng)前的唯一線程,從隊(duì)列取任務(wù),執(zhí)行完一個(gè),再繼續(xù)取,一個(gè)人(一條線程)夜以繼日地干活。
使用場景
適用于串行執(zhí)行任務(wù)的場景,一個(gè)任務(wù)一個(gè)任務(wù)地執(zhí)行。
newScheduledThreadPool 定時(shí)及周期執(zhí)行的線程池
public ScheduledThreadPoolExecutor(int corePoolSize) { super(corePoolSize, Integer.MAX_VALUE, 0, NANOSECONDS, new DelayedWorkQueue()); }
線程池特點(diǎn)
最大線程數(shù)為Integer.MAX_VALUE
阻塞隊(duì)列是DelayedWorkQueue
keepAliveTime為0
scheduleAtFixedRate() :按某種速率周期執(zhí)行
scheduleWithFixedDelay():在某個(gè)延遲后執(zhí)行
工作機(jī)制
- 添加一個(gè)任務(wù)
- 線程池中的線程從 DelayQueue 中取任務(wù)
- 線程從 DelayQueue 中獲取 time 大于等于當(dāng)前時(shí)間的task
- 執(zhí)行完后修改這個(gè) task 的 time 為下次被執(zhí)行的時(shí)間
- 這個(gè) task 放回DelayQueue隊(duì)列中
使用場景
周期性執(zhí)行任務(wù)的場景,需要限制線程數(shù)量的場景
以上就是java Executors常用線程池的詳細(xì)內(nèi)容,更多關(guān)于java Executors線程池的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
Java實(shí)現(xiàn)隨機(jī)生成大小寫混合的卡密的步驟
在現(xiàn)代軟件開發(fā)中,生成隨機(jī)卡密是一個(gè)常見的需求,尤其是在需要為用戶生成唯一識(shí)別碼或安全令牌的場景中,卡密通常由數(shù)字和字母組成,有時(shí)還會(huì)包含特殊字符,本文通過代碼講解的非常詳細(xì),需要的朋友可以參考下2024-11-11通過反射注解批量插入數(shù)據(jù)到DB的實(shí)現(xiàn)方法
今天小編就為大家分享一篇關(guān)于通過反射注解批量插入數(shù)據(jù)到DB的實(shí)現(xiàn)方法,小編覺得內(nèi)容挺不錯(cuò)的,現(xiàn)在分享給大家,具有很好的參考價(jià)值,需要的朋友一起跟隨小編來看看吧2019-03-03解析如何用兩個(gè)棧來實(shí)現(xiàn)隊(duì)列的方法
本篇文章是對(duì)如何用兩個(gè)棧實(shí)現(xiàn)隊(duì)列的方法進(jìn)行了詳細(xì)的分析介紹,需要的朋友參考下2013-06-06Java SE使用數(shù)組實(shí)現(xiàn)高速數(shù)字轉(zhuǎn)換功能
隨著大數(shù)據(jù)時(shí)代的到來,數(shù)字轉(zhuǎn)換功能變得越來越重要,在Java開發(fā)中,數(shù)字轉(zhuǎn)換功能也是經(jīng)常用到的,下面我們就來學(xué)習(xí)一下如何使用Java SE數(shù)組實(shí)現(xiàn)高速的數(shù)字轉(zhuǎn)換功能吧2023-11-11java實(shí)現(xiàn)頁面多查詢條件必選的統(tǒng)一處理思路
這篇文章主要為大家介紹了java實(shí)現(xiàn)頁面多查詢條件必選的統(tǒng)一處理思路詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-06-06Java獲取支付寶OpenID的實(shí)現(xiàn)方法
在Java中,通過支付寶開放平臺(tái)API可以獲取用戶的OpenID,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2024-09-09mybatis類型轉(zhuǎn)換器如何實(shí)現(xiàn)數(shù)據(jù)加解密
這篇文章主要介紹了mybatis類型轉(zhuǎn)換器如何實(shí)現(xiàn)數(shù)據(jù)加解密,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2021-09-09