原來你是這樣的SpringBoot--Async非同步任務

来源:https://www.cnblogs.com/janes/archive/2023/08/25/17648727.html
-Advertisement-
Play Games

本節我們一起學習一下SpringBoot中的非同步調用,主要用於優化耗時較長的操作,提高系統性能和吞吐量。 # 一、新建項目,啟動非同步調用 首先給啟動類增加註解@EnableAsync,支持非同步調用 ``` @EnableAsync @SpringBootApplication public clas ...


本節我們一起學習一下SpringBoot中的非同步調用,主要用於優化耗時較長的操作,提高系統性能和吞吐量。

一、新建項目,啟動非同步調用

首先給啟動類增加註解@EnableAsync,支持非同步調用

@EnableAsync
@SpringBootApplication
public class CathySpringbootDemoApplication {

    public static void main(String[] args) {
        SpringApplication.run(CathySpringbootDemoApplication.class, args);
    }

}

然後定義要執行的Task,分類增加一個同步方法和非同步方法,其中非同步方法需要增加註解@Async

@Component
public class AsyncTask {
    /**
     * 非同步任務,需要註解@Async
     *
     * @param taskId 任務編號id
     * @param second 執行時長,模擬慢任務
     * @return
     */
    @Async
    public Future<Boolean> asyncExec(int taskId, Long second) {
        exec(taskId, second);
        return new AsyncResult<>(Boolean.TRUE);
    }

    public void exec(int taskId, Long second) {
        System.out.println("開始執行任務" + taskId);
        try {
            Thread.sleep(second * 1000);
        } catch (InterruptedException e) {
            throw new RuntimeException(e);
        }
        System.out.println("結束執行任務" + taskId);
    }
}

其實接下來就可以在controller中創建介面來進行簡單的測試了

@RestController
@RequestMapping("/async")
public class AsyncController {
    @Autowired
    AsyncTask asyncTask;

    @GetMapping("sync_task")
    public String syncTask() {
        long start = System.currentTimeMillis();
        asyncTask.exec(1, 3L);
        asyncTask.exec(2, 3L);
        asyncTask.exec(3, 3L);
        long time = System.currentTimeMillis() - start;
        return "同步執行,耗時" + time;
    }

    @GetMapping("async_task")
    public String asyncTask() {
        long start = System.currentTimeMillis();
        Future<Boolean> f1 = asyncTask.asyncExec(1, 3L);
        Future<Boolean> f2 = asyncTask.asyncExec(2, 3L);
        Future<Boolean> f3 = asyncTask.asyncExec(3, 3L);
        try {
            f1.get();
            f2.get();
            f3.get();
        } catch (InterruptedException e) {
            throw new RuntimeException(e);
        } catch (ExecutionException e) {
            throw new RuntimeException(e);
        }

        long time = System.currentTimeMillis() - start;
        return "非同步執行,耗時" + time;
    }
}

啟動程式,查看介面響應結果:
http://localhost:16001/async/sync_task
img

http://localhost:16001/async/async_task
img

註意:非同步方法和調用一定要寫在不同的類中

二、線程池配置

上面的例子,在耗時服務多的情況下,使用非同步方法確實提高了響應速度。但是它預設啟用的是Spring預設的線程池SimpleAsyncTaskExecutor,不太靈活。我們把非同步請求多增加幾次調用看看效果:

@GetMapping("async_task")
    public String asyncTask() {
        long start = System.currentTimeMillis();
        List<Future<Boolean>> list = new ArrayList<>();
        for (int i = 0; i < 20; i++) {
            Future<Boolean> fi = asyncTask.asyncExec(i, 10L);
            list.add(fi);
        }
        for (int i = 0; i < 20; i++) {

            list.forEach(x -> {
                try {
                    x.get();
                } catch (InterruptedException e) {
                    throw new RuntimeException(e);
                } catch (ExecutionException e) {
                    throw new RuntimeException(e);
                }
            });
        }

        long time = System.currentTimeMillis() - start;
        return "非同步執行,耗時" + time;
    }

img
從上面的運行效果來看,一旦超過8個並行執行的任務,就開始出現等待了。

接下來,我們自定義線程池

@Bean
 public TaskExecutor threadPoolTaskExecutor(){
     ThreadPoolTaskExecutor executor = new ThreadPoolTaskExecutor();
     executor.setCorePoolSize(8);
     executor.setMaxPoolSize(16);
     executor.setQueueCapacity(20);
     executor.setKeepAliveSeconds(30);
     executor.setWaitForTasksToCompleteOnShutdown(true);
     executor.setThreadNamePrefix("task-thread-");
     executor.setRejectedExecutionHandler(new ThreadPoolExecutor.DiscardOldestPolicy());

     executor.initialize();
     return executor;
 }

然後在非同步方法的註解中,明確指定所使用的線程池

@Async("threadPoolTaskExecutor")
    public Future<Boolean> asyncExec(int taskId, Long second) {
        exec(taskId, second);
        return new AsyncResult<>(Boolean.TRUE);
    }

執行效果如下:

img

img

可以看出,線程池設置的參數已經生效。


本人公眾號[ 敬YES ]同步更新,歡迎大家關註~

img

作者:陳敬(公眾號:敬姐嘚啵嘚)
出處:http://www.cnblogs.com/janes/
博客文章僅供交流學習,請勿用於商業用途。如需轉載,請務必註明出處。


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • **塑造軟體新生態,賦能發展新變革。** **8月31日-9月2日**,第二十五屆**中國國際軟體博覽會**將於天津梅江會展中心召開。本屆軟博會由中國電子信息行業聯合會主辦,聚焦全球軟體前沿技術與產業發展方向,充分展示軟體賦能數字經濟、信息技術應用創新、工業互聯網平臺、智能製造及元宇宙等多領域發展成 ...
  • 這裡給大家分享我在網上總結出來的一些知識,希望對大家有所幫助 最近有一個需求,要我實現一個動畫效果,效果如下 簡單分析了一下效果,是一個3d的效果,首先是一個圓,接著是兩段圓環,第三層是一堆小圓環,最裡面是一些線上運動,有著漸變色的矩形。 第一層的圓環很簡單。 第二層的圓環其實也挺簡單的,只要在設置 ...
  • 當涉及到優化 Flutter 應用時,考慮性能、UI 渲染和記憶體管理是至關重要的。在本篇文章中,我們將通過實例深入討論這些主題,展示如何通過優化技巧改進你的 Flutter 應用。 ...
  • >我們是[袋鼠雲數棧 UED 團隊](http://ued.dtstack.cn/),致力於打造優秀的一站式數據中台產品。我們始終保持工匠精神,探索前端道路,為社區積累並傳播經驗價值。 >本文作者:佳嵐 ### 前言 `Cookie`實際上是一小段的文本信息,它產生的原因是由於HTTP 協議是**無 ...
  • # 模板頁的重要性 Vue 項目中使用佈局組件來創建頁面佈局的方式是完全可行的,而且在很多項目中都被廣泛採用,包括像 ruoyi 這樣的框架。這種模式有助於實現統一的頁面佈局結構,減少重覆代碼,並提高代碼的可維護性。 讓我們具體分析一下你提到的 ruoyi 框架的做法: 1. **Layout 組件 ...
  • 環境: SpringBoot2.7.8 背景: 在增加出庫訂單時需要對物品表的的數量進行修改 因此我在OutboundController中創建了幾個公共方法,並將其註入到Spring中,結果給我報了這一串錯誤。 Description:The dependencies of some of the ...
  • [TOC] 本文主要介紹ImGui應用中的一些界面優化方法,如果是第一次使用ImGui推薦從上一篇文章開始:[使用C++界面框架ImGUI開發一個簡單程式](https://www.cnblogs.com/timefiles/p/17632348.html),最終的界面效果如下: ![image]( ...
  • ## 所有類的基類 Object Lua 沒有嚴格的 oo(Object-Oriented)定義,可以利用元表特性來實現 先定義所有類的基類,即`Object`類。代碼順序從上到下,自成一體。[完整代碼](#oo.lua) 定義一個空表 `Object` ,`__index` 指向其自身(繼承將直接 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...