Spring中的ThreadPoolTaskExecutor線程池使用詳解
一、配置
1.1 將 ThreadPoolTaskExecutor 注入到 spring 容器內(nèi)
@Configuration public class ThreadTaskPoolExecutorConfiguration { @Bean public ThreadPoolTaskExecutor threadPoolTaskExecutor(){ ThreadPoolTaskExecutor taskExecutor = new ThreadPoolTaskExecutor(); // 核心線程數(shù) taskExecutor.setCorePoolSize(5); // 最大線程數(shù) taskExecutor.setMaxPoolSize(15); // 隊(duì)列大小 默認(rèn)使用LinkedBlockingQueue taskExecutor.setQueueCapacity(100); // 線程最大空閑時(shí)間 taskExecutor.setKeepAliveSeconds(300); // 拒絕策略 默認(rèn)new ThreadPoolExecutor.AbortPolicy() taskExecutor.setRejectedExecutionHandler(new ThreadPoolExecutor.CallerRunsPolicy()); // 線程名稱前綴 taskExecutor.setThreadNamePrefix("My-Task-Executor-"); //交給spring托管的會(huì)自動(dòng)初始化,因?yàn)閷?shí)現(xiàn)了InitializingBean接口 // taskExecutor.initialize(); return taskExecutor; } }
1.2 拒絕策略配置
rejectedExecutionHandler 字段用于配置拒絕策略,常用的拒絕策略如下:
- AbortPolicy:用于被拒絕任務(wù)的處理程序,它將拋出 RejectedExecutionException。
- CallerRunsPolicy:用于被拒絕任務(wù)的處理程序,它直接在 execute 方法的調(diào)用線程中運(yùn)行被拒絕的任務(wù)。
- DiscardOldestPolicy:用于被拒絕任務(wù)的處理程序,它放棄最舊的未處理請(qǐng)求,然后重試 execute。
- DiscardPolicy:用于被拒絕任務(wù)的處理程序,默認(rèn)情況下它將丟棄被拒絕的任務(wù)。
其他說(shuō)明:
- 為了實(shí)現(xiàn)某些特殊的業(yè)務(wù)需求,用戶可以選擇使用自定義策略,只需實(shí)現(xiàn)RejectedExecutionHandler接口即可。
- 建議配置threadNamePrefix屬性,出問(wèn)題時(shí)可以更方便的進(jìn)行排查。
1.3 配置線程池個(gè)數(shù)
- 如果是 CPU 密集型任務(wù),那么線程池的線程個(gè)數(shù)應(yīng)該盡量少一些,一般為 CPU 的個(gè)數(shù)+1條線程。
- 如果是 IO 密集型任務(wù),那么線程池的線程可以放的很大,如 2*CPU 的個(gè)數(shù)。
- 對(duì)于混合型任務(wù),如果可以拆分的話,通過(guò)拆分成 CPU 密集型和 IO 密集型兩種來(lái)提高執(zhí)行效率;如果不能拆分的的話就可以根據(jù)實(shí)際情況來(lái)調(diào)整線程池中線程的個(gè)數(shù)
二、處理流程
- 當(dāng)一個(gè)任務(wù)被提交到線程池時(shí),首先查看線程池的核心線程是否都在執(zhí)行任務(wù),否就選擇一條線程執(zhí)行任務(wù),是就執(zhí)行第二步。
- 查看核心線程池是否已滿,不滿就創(chuàng)建一條線程執(zhí)行任務(wù),否則執(zhí)行第三步。
- 查看任務(wù)隊(duì)列是否已滿,不滿就將任務(wù)存儲(chǔ)在任務(wù)隊(duì)列中,否則執(zhí)行第四步。
- 查看線程池是否已滿,不滿就創(chuàng)建一條線程執(zhí)行任務(wù),否則就按照策略處理無(wú)法執(zhí)行的任務(wù)。
在 ThreadPoolExecutor 中表現(xiàn)為:
- 如果當(dāng)前運(yùn)行的線程數(shù)小于corePoolSize,那么就創(chuàng)建線程來(lái)執(zhí)行任務(wù)(執(zhí)行時(shí)需要獲取全局鎖)。
- 如果運(yùn)行的線程大于或等于corePoolSize,那么就把task加入BlockQueue。
- 如果創(chuàng)建的線程數(shù)量大于BlockQueue的最大容量,那么創(chuàng)建新線程來(lái)執(zhí)行該任務(wù)。
- 如果創(chuàng)建線程導(dǎo)致當(dāng)前運(yùn)行的線程數(shù)超過(guò)maximumPoolSize,就根據(jù)飽和策略來(lái)拒絕該任務(wù)。
三、關(guān)閉線程池
調(diào)用shutdown或者shutdownNow,兩者都不會(huì)接受新的任務(wù),而且通過(guò)調(diào)用要停止線程的interrupt方法來(lái)中斷線程,有可能線程永遠(yuǎn)不會(huì)被中斷,不同之處在于shutdownNow會(huì)首先將線程池的狀態(tài)設(shè)置為STOP,然后嘗試停止所有線程(有可能導(dǎo)致部分任務(wù)沒(méi)有執(zhí)行完)然后返回未執(zhí)行任務(wù)的列表。而shutdown則只是將線程池的狀態(tài)設(shè)置為shutdown,然后中斷所有沒(méi)有執(zhí)行任務(wù)的線程,并將剩余的任務(wù)執(zhí)行完。
四、監(jiān)控線程池狀態(tài)
常用狀態(tài):
- taskCount:線程需要執(zhí)行的任務(wù)個(gè)數(shù)。
- completedTaskCount:線程池在運(yùn)行過(guò)程中已完成的任務(wù)數(shù)。
- largestPoolSize:線程池曾經(jīng)創(chuàng)建過(guò)的最大線程數(shù)量。
- getPoolSize獲取當(dāng)前線程池的線程數(shù)量。
- getActiveCount:獲取活動(dòng)的線程的數(shù)量
通過(guò)繼承線程池,重寫beforeExecute,afterExecute 和 terminated 方法來(lái)在線程執(zhí)行任務(wù)前,線程執(zhí)行任務(wù)結(jié)束,和線程終結(jié)前獲取線程的運(yùn)行情況,根據(jù)具體情況調(diào)整線程池的線程數(shù)量。
五、實(shí)戰(zhàn)
@RunWith(SpringRunner.class) @SpringBootTest(classes = TestApplication.class) public class TestApplicationTests { // 注入ThreadPoolTaskExecutor @Resource private ThreadPoolTaskExecutor threadPoolTaskExecutor; @Test public void ThreadTest(){ System.out.println(threadPoolTaskExecutor); System.out.println("new Runnable()"); // 創(chuàng)建并執(zhí)行線程,方式一 threadPoolTaskExecutor.execute(new Runnable() { @Override public void run() { for (int i = 0; i < 100; i++) { System.out.println("new Runnable()"+i+"當(dāng)前線程"+Thread.currentThread().getName()); } } }); // // 創(chuàng)建并執(zhí)行線程,方式二 System.out.println("lambda"); threadPoolTaskExecutor.execute(() -> { for (int i = 0; i < 100; i++) { System.out.println("lambda"+i+"當(dāng)前線程"+Thread.currentThread().getName()); } }); } }
運(yùn)行截圖:
用lambda表達(dá)式實(shí)現(xiàn)Runnable
我開(kāi)始使用Java 8時(shí),首先做的就是使用lambda表達(dá)式替換匿名類,而實(shí)現(xiàn)Runnable接口是匿名類的最好示例。
看一下Java 8之前的runnable實(shí)現(xiàn)方法,需要4行代碼,而使用lambda表達(dá)式只需要一行代碼。我們?cè)谶@里做了什么呢?那就是用() -> {}代碼塊替代了整個(gè)匿名類。
// Java 8之前: new Thread(new Runnable() { @Override public void run() { System.out.println("Before Java8, too much code for too little to do"); } }).start(); //Java 8方式: new Thread( () -> System.out.println("In Java8, Lambda expression rocks !!") ).start();
到此這篇關(guān)于Spring中的ThreadPoolTaskExecutor線程池使用詳解的文章就介紹到這了,更多相關(guān)ThreadPoolTaskExecutor線程池內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Mybatis內(nèi)置參數(shù)之_parameter和_databaseId的使用
這篇文章主要介紹了Mybatis內(nèi)置參數(shù)之_parameter和_databaseId的使用方式,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2023-12-12springboot+mybatis如何屏蔽掉mybatis日志
這篇文章主要介紹了springboot+mybatis如何屏蔽掉mybatis日志問(wèn)題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2023-05-05java實(shí)現(xiàn)基于UDP協(xié)議網(wǎng)絡(luò)Socket編程(C/S通信)
這篇文章主要介紹了java實(shí)現(xiàn)基于UDP協(xié)議網(wǎng)絡(luò)Socket編程(C/S通信),文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2020-10-10SpringBoot如何基于POI-tl和word模板導(dǎo)出龐大的Word文件
這篇文章主要介紹了SpringBoot如何基于POI-tl和word模板導(dǎo)出龐大的Word文件,poi-tl是一個(gè)基于Apache?POI的Word模板引擎,也是一個(gè)免費(fèi)開(kāi)源的Java類庫(kù)2022-08-08java時(shí)間戳與日期相互轉(zhuǎn)換工具詳解
這篇文章主要為大家詳細(xì)介紹了java各種時(shí)間戳與日期之間相互轉(zhuǎn)換的工具,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2017-12-12java實(shí)現(xiàn)jdbc查詢結(jié)果集result轉(zhuǎn)換成對(duì)應(yīng)list集合
本文給大家匯總介紹了java實(shí)現(xiàn)jdbc查詢結(jié)果集result轉(zhuǎn)換成對(duì)應(yīng)list集合,十分的簡(jiǎn)單,有相同需求的小伙伴可以參考下。2015-12-12