衡阳派盒市场营销有限公司

0
  • 聊天消息
  • 系統消息
  • 評論與回復
登錄后你可以
  • 下載海量資料
  • 學習在線課程
  • 觀看技術視頻
  • 寫文章/發帖/加入社區
會員中心
創作中心

完善資料讓更多小伙伴認識你,還能領取20積分哦,立即完善>

3天內不再提示

Spring 的線程池應用

科技綠洲 ? 來源:Java技術指北 ? 作者:Java技術指北 ? 2023-10-13 10:47 ? 次閱讀

我們在日常開發中,經常跟多線程打交道,Spring 為我們提供了一個線程池方便我們開發,它就是 ThreadPoolTaskExecutor ,接下來我們就來聊聊 Spring 的線程池吧。


使用@Async聲明多線程

SpringBoot 提供了注解 @Async 來使用線程池, 具體使用方法如下:

  1. 在啟動類(配置類)添加@EnableAsync來開啟線程池
  2. 在需要開啟子線程的方法上添加注解 @Async

下面是一個簡單的例子:

@Component
@EnableAsync
@EnableScheduling
public class ScheduleTask {

    @Async
    @Scheduled(fixedRate = 2000)
    public void testAsync1() {
        try {
            Thread.sleep(6000);
            System.out.println(LocalDateTime.now() + "--線程1:" + Thread.currentThread().getName());
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
    }

    @Async
    @Scheduled(cron = "*/2 * * * * ?")
    public void testAsync2() {
        try {
            Thread.sleep(1000);
            System.out.println(LocalDateTime.now() + "--線程2:" + Thread.currentThread().getName());
        } catch (Exception ex) {
            ex.printStackTrace();
        }
    }
}

啟動項目,得到如下日志結果:

圖片

可以發現在當前環境下 task-${id} 這個 id 并不是一直增長的,而是一直在復用 1-8。這個時候可能就會有的小伙伴們會比較好奇,默認的不是 SimpleAsyncTaskExecutor 嗎?為什么從日志打印的效果上看像是一直在復用 8 個線程,難道用的是 ThreadPoolTaskExecutor?

原因是 SpringBoot2.1.0 版本后,新增了 TaskExecutionAutoConfiguration 配置類。其中聲明的默認線程池就是 ThreadPoolTaskExecutor 。而 @Async 在選擇執行器的時候會先去 IOC 容器中先找是否有 TaskExecutor 的 Bean對象,所以在當前版本 SpringBoot 中,@Async 的默認 TaskExecutor 是 ThreadPoolTaskExecutor。

線程池配置

在 SpringBoot 項目中,我們可以在 yaml 或者 properties 配置文件中配置,或者使用 @Configuration 配置,下面演示配置方法。

  1. application.properties配置文件中配置
# 核心線程池數
spring.task.execution.pool.core-size=5
# 最大線程池數
spring.task.execution.pool.max-size=10
# 任務隊列的容量
spring.task.execution.pool.queue-capacity=5
# 非核心線程的存活時間
spring.task.execution.pool.keep-alive=60
# 線程池的前綴名稱
spring.task.execution.thread-name-prefix=test-task-
  1. 配置類中配置
@Bean(name = "myThreadPoolTaskExecutor")
public ThreadPoolTaskExecutor getMyThreadPoolTaskExecutor() {
    ThreadPoolTaskExecutor taskExecutor = new ThreadPoolTaskExecutor();
    int i = Runtime.getRuntime().availableProcessors();
    taskExecutor.setCorePoolSize(i * 2);
    taskExecutor.setMaxPoolSize(i * 2);
    taskExecutor.setQueueCapacity(i * 2 * 100);
    taskExecutor.setKeepAliveSeconds(60);
    taskExecutor.setThreadNamePrefix("my-task-");
    taskExecutor.initialize();
    return taskExecutor;
}

拒絕策略

RejectedExectutionHandler 參數字段用于配置絕策略,常用拒絕策略如下

  • AbortPolicy:用于被拒絕任務的處理程序,它將拋出RejectedExecutionException
  • CallerRunsPolicy:用于被拒絕任務的處理程序,它直接在execute方法的調用線程中運行被拒絕的任務。
  • DiscardOldestPolicy:用于被拒絕任務的處理程序,它放棄最舊的未處理請求,然后重試execute。
  • DiscardPolicy:用于被拒絕任務的處理程序,默認情況下它將丟棄被拒絕的任務。

處理流程

  1. 查看核心線程池是否已滿,不滿就創建一條線程執行任務,否則執行第2步。
  2. 查看任務隊列是否已滿,不滿就將任務存儲在任務隊列中,否則執行第3步。
  3. 查看線程池是否已滿,即就是是否達到最大線程池數,不滿就創建一條線程執行任務,否則就按照策略處理無法執行的任務。

使用注意

  1. 注解的方法必須是 public 方法。
  2. 方法一定要從另一個類中調用,也就是從類的外部調用,類的內部調用是無效的,因為 @Transactional 和 @Async 注解的實現都是基于 Spring 的 AOP ,而 AOP 的實現是基于動態代理模式實現的。那么注解失效的原因就很明顯了,有可能因為調用方法的是對象本身而不是代理對象,因為沒有經過 Spring 容器。
  3. 異步方法使用注解 @Async 的返回值只能為 void 或者 Future。

總結

上面簡單介紹了 Spring 自帶的線程池 ThreadPoolTaskExecutor 的配置和使用,并且講了線程池的參數和處理流程。當然Spring提供了7個線程池的實現,感興趣的可以自行了解~

聲明:本文內容及配圖由入駐作者撰寫或者入駐合作網站授權轉載。文章觀點僅代表作者本人,不代表電子發燒友網立場。文章及其配圖僅供工程師學習之用,如有內容侵權或者其他違規問題,請聯系本站處理。 舉報投訴
  • 文件
    +關注

    關注

    1

    文章

    570

    瀏覽量

    24822
  • 容器
    +關注

    關注

    0

    文章

    499

    瀏覽量

    22120
  • spring
    +關注

    關注

    0

    文章

    340

    瀏覽量

    14388
  • 線程池
    +關注

    關注

    0

    文章

    57

    瀏覽量

    6893
收藏 人收藏

    評論

    相關推薦

    跨平臺的線程組件--TP組件

    /銷毀代價是很高的。那么我們要怎么去設計多線程編程呢???答案:對于長駐的線程,我們可以創建獨立的線程去執行。但是非長駐的線程,我們可以通過線程
    的頭像 發表于 04-06 15:39 ?913次閱讀

    Java中的線程包括哪些

    線程是用來統一管理線程的,在 Java 中創建和銷毀線程都是一件消耗資源的事情,線程可以重復
    的頭像 發表于 10-11 15:33 ?858次閱讀
    Java中的<b class='flag-5'>線程</b><b class='flag-5'>池</b>包括哪些

    線程是如何實現的

    線程的概念是什么?線程是如何實現的?
    發表于 02-28 06:20

    基于線程技術集群接入點的應用研究

    本文在深入研究高級線程技術的基礎上,分析、研究了固定線程數目的線程線程數目動態變化的
    發表于 01-22 14:21 ?5次下載

    如何正確使用SpringBoot中的線程

    來自丨CSDN https://blog.csdn.net/m0_37701381/article/details/81072774 使用步驟 先創建一個線程的配置,讓Spring Boot加載
    的頭像 發表于 09-02 17:14 ?2129次閱讀

    基于Nacos的簡單動態化線程實現

    本文以Nacos作為服務配置中心,以修改線程核心線程數、最大線程數為例,實現一個簡單的動態化線程
    發表于 01-06 14:14 ?890次閱讀

    線程線程

    線程通常用于服務器應用程序。 每個傳入請求都將分配給線程池中的一個線程,因此可以異步處理請求,而不會占用主線程,也不會延遲后續請求的處理
    的頭像 發表于 02-28 09:53 ?832次閱讀
    多<b class='flag-5'>線程</b>之<b class='flag-5'>線程</b><b class='flag-5'>池</b>

    Java線程核心原理

    看過Java線程源碼的小伙伴都知道,在Java線程池中最核心的類就是ThreadPoolExecutor,
    的頭像 發表于 04-21 10:24 ?908次閱讀

    細數線程的10個坑

    JDK開發者提供了線程的實現類,我們基于Executors組件,就可以快速創建一個線程
    的頭像 發表于 06-16 10:11 ?765次閱讀
    細數<b class='flag-5'>線程</b><b class='flag-5'>池</b>的10個坑

    線程線程怎么釋放

    線程分組看,pool名開頭線程占616條,而且waiting狀態也是616條,這個點就非常可疑了,我斷定就是這個pool開頭線程導致的問題。我們先排查為何這個
    發表于 07-31 10:49 ?2349次閱讀
    <b class='flag-5'>線程</b><b class='flag-5'>池</b>的<b class='flag-5'>線程</b>怎么釋放

    線程基本概念與原理

    一、線程基本概念與原理 1.1 線程概念及優勢 C++線程簡介
    的頭像 發表于 11-10 10:24 ?578次閱讀

    線程的基本概念

    線程的基本概念 不管線程是什么東西!但是我們必須知道線程被搞出來的目的就是:提高程序執行效
    的頭像 發表于 11-10 16:37 ?560次閱讀
    <b class='flag-5'>線程</b><b class='flag-5'>池</b>的基本概念

    線程七大核心參數執行順序

    線程是一種用于管理和調度線程執行的技術,通過將任務分配到線程池中的線程進行處理,可以有效地控制并發線程
    的頭像 發表于 12-04 16:45 ?1145次閱讀

    線程的創建方式有幾種

    線程是一種用于管理和調度線程的技術,能夠有效地提高系統的性能和資源利用率。它通過預先創建一組線程并維護一個工作隊列,將任務提交給線程
    的頭像 發表于 12-04 16:52 ?933次閱讀

    什么是動態線程?動態線程的簡單實現思路

    因此,動態可監控線程一種針對以上痛點開發的線程管理工具。主要可實現功能有:提供對 Spring 應用內
    的頭像 發表于 02-28 10:42 ?722次閱讀
    奥斯卡百家乐官网的玩法技巧和规则| 网上赌百家乐官网的玩法技巧和规则 | 南京百家乐电| 太阳城百家乐优惠| 杨公风水24山| 汕尾市| 百家乐佛泰阁| 八大胜百家乐官网的玩法技巧和规则| 玩百家乐官网犯法| 百家乐官网视频游戏官网| 百家乐官网视频游戏帐号| 门赌场百家乐官网的规则| 百家乐官网娱乐城7| 百家乐官网单双打法| 百家乐官网能赢到钱吗| 百家乐官网平注法是什么| 百家乐官网网络赌博网址| 百家乐官网投注平台信誉排名| 百家乐官网的必胜方法| 澳门百家乐官网家用保险柜| 百家乐官网赌场博彩赌场网| 百家乐官网娱乐网开户| 百家乐官网技巧平注常赢法| 百家乐视频大厅| 怎样看百家乐路单| 华盛顿百家乐的玩法技巧和规则| 什么是百家乐赌博| 百家乐平玩法官方网址| 大发888官方网站登录| 大发888心水论坛| 大发888 casino组件下载| 拉萨市| 足球百家乐官网投注计算| 百家乐官网技巧技巧| 百家乐游戏玩法规则| 博彩百家乐组选六六组| 大发888是什么东| 百家乐官网下注几多| 木棉百家乐官网网络| 最佳场百家乐官网的玩法技巧和规则 | 开16个赌场敛财|