亚洲乱码中文字幕综合,中国熟女仑乱hd,亚洲精品乱拍国产一区二区三区,一本大道卡一卡二卡三乱码全集资源,又粗又黄又硬又爽的免费视频

Spring中的ThreadPoolTaskExecutor線程池使用詳解

 更新時(shí)間:2024年01月31日 10:19:17   作者:卷不動(dòng)躺不平的粥  
這篇文章主要介紹了Spring中的ThreadPoolTaskExecutor線程池使用詳解,ThreadPoolTaskExecutor 是 Spring框架提供的一個(gè)線程池實(shí)現(xiàn),用于管理和執(zhí)行多線程任務(wù),它是TaskExecutor接口的實(shí)現(xiàn),提供了在 Spring 應(yīng)用程序中創(chuàng)建和配置線程池的便捷方式,需要的朋友可以參考下

一、配置

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ù)

二、處理流程

  1. 當(dāng)一個(gè)任務(wù)被提交到線程池時(shí),首先查看線程池的核心線程是否都在執(zhí)行任務(wù),否就選擇一條線程執(zhí)行任務(wù),是就執(zhí)行第二步。
  2. 查看核心線程池是否已滿,不滿就創(chuàng)建一條線程執(zhí)行任務(wù),否則執(zhí)行第三步。
  3. 查看任務(wù)隊(duì)列是否已滿,不滿就將任務(wù)存儲(chǔ)在任務(wù)隊(duì)列中,否則執(zhí)行第四步。
  4. 查看線程池是否已滿,不滿就創(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)文章

  • 基于JAVA中的四種JSON解析方式詳解

    基于JAVA中的四種JSON解析方式詳解

    這篇文章主要介紹了基于JAVA中的四種JSON解析方式詳解,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧
    2019-09-09
  • Mybatis內(nèi)置參數(shù)之_parameter和_databaseId的使用

    Mybatis內(nèi)置參數(shù)之_parameter和_databaseId的使用

    這篇文章主要介紹了Mybatis內(nèi)置參數(shù)之_parameter和_databaseId的使用方式,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2023-12-12
  • Protobuf的簡(jiǎn)要介紹及使用詳解

    Protobuf的簡(jiǎn)要介紹及使用詳解

    這篇文章主要介紹了Protobuf的簡(jiǎn)要介紹及使用,本文給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下
    2020-10-10
  • springboot+mybatis如何屏蔽掉mybatis日志

    springboot+mybatis如何屏蔽掉mybatis日志

    這篇文章主要介紹了springboot+mybatis如何屏蔽掉mybatis日志問(wèn)題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2023-05-05
  • SpringMVC中文亂碼踩坑記錄

    SpringMVC中文亂碼踩坑記錄

    這篇文章主要介紹了SpringMVC中文亂碼踩坑記錄,本文通過(guò)實(shí)例代碼給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下
    2021-08-08
  • Java綜合整理堆排序?快速排序?歸并排序

    Java綜合整理堆排序?快速排序?歸并排序

    堆排序是利用堆這種數(shù)據(jù)結(jié)構(gòu)而設(shè)計(jì)的一種排序算法,堆排序是一種選擇排序,它的最壞,最好,平均時(shí)間復(fù)雜度均為O(nlogn),它也是不穩(wěn)定排序。首先簡(jiǎn)單了解下堆結(jié)構(gòu)
    2022-01-01
  • java實(shí)現(xiàn)基于UDP協(xié)議網(wǎng)絡(luò)Socket編程(C/S通信)

    java實(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-10
  • SpringBoot如何基于POI-tl和word模板導(dǎo)出龐大的Word文件

    SpringBoot如何基于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-08
  • java時(shí)間戳與日期相互轉(zhuǎn)換工具詳解

    java時(shí)間戳與日期相互轉(zhuǎn)換工具詳解

    這篇文章主要為大家詳細(xì)介紹了java各種時(shí)間戳與日期之間相互轉(zhuǎn)換的工具,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2017-12-12
  • java實(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集合

    本文給大家匯總介紹了java實(shí)現(xiàn)jdbc查詢結(jié)果集result轉(zhuǎn)換成對(duì)應(yīng)list集合,十分的簡(jiǎn)單,有相同需求的小伙伴可以參考下。
    2015-12-12

最新評(píng)論