欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

詳解springboot通過Async注解實現(xiàn)異步任務(wù)及回調(diào)的方法

 更新時間:2022年05月11日 10:05:58   作者:字母哥哥  
這篇文章主要介紹了springboot通過Async注解實現(xiàn)異步任務(wù)及回調(diào),文中通過一個簡單示例來直觀的理解什么是同步調(diào)用,在單元測試用例中,注入?SyncTask?對象,并在測試用例中執(zhí)行?doTaskOne(),doTaskTwo(),doTaskThree()?三個方法,具體實現(xiàn)方式跟隨小編一起看看吧

前言

什么是異步調(diào)用?

異步調(diào)用是相對于同步調(diào)用而言的,同步調(diào)用是指程序按預(yù)定順序一步步執(zhí)行,每一步必須等到上一步執(zhí)行完后才能執(zhí)行,異步調(diào)用則無需等待上一步程序執(zhí)行完即可執(zhí)行。異步調(diào)用可以減少程序執(zhí)行時間。

1. 環(huán)境準(zhǔn)備

在這里插入圖片描述

Spring Boot 入口類上配置 @EnableAsync 注解開啟異步處理。
創(chuàng)建任務(wù)抽象類 AbstractTask,并實現(xiàn)三個任務(wù)方法 doTaskOne(),doTaskTwo(),doTaskThree()。

public abstract class AbstractTask {
    private static Random random = new Random();
    public void doTaskOne() throws Exception {
        System.out.println("開始做任務(wù)一");
        long start = currentTimeMillis();
        sleep(random.nextInt(10000));
        long end = currentTimeMillis();
        System.out.println("完成任務(wù)一,耗時:" + (end - start) + "毫秒");
    }
    public void doTaskTwo() throws Exception {
        System.out.println("開始做任務(wù)二");
        long start = currentTimeMillis();
        sleep(random.nextInt(10000));
        long end = currentTimeMillis();
        System.out.println("完成任務(wù)二,耗時:" + (end - start) + "毫秒");
    }
    public void doTaskThree() throws Exception {
        System.out.println("開始做任務(wù)三");
        long start = currentTimeMillis();
        sleep(random.nextInt(10000));
        long end = currentTimeMillis();
        System.out.println("完成任務(wù)三,耗時:" + (end - start) + "毫秒");
    }
}

2. 同步調(diào)用

下面通過一個簡單示例來直觀的理解什么是同步調(diào)用:

定義 Task 類,繼承 AbstractTask,三個處理函數(shù)分別模擬三個執(zhí)行任務(wù)的操作,操作消耗時間隨機取(10 秒內(nèi))。

@Component
public class SyncTask extends AbstractTask {
}

單元測試 用例中,注入 SyncTask 對象,并在測試用例中執(zhí)行 doTaskOne(),doTaskTwo()doTaskThree() 三個方法。

@RunWith(SpringRunner.class)
@SpringBootTest
public class TaskTest {
    @Autowired
    private SyncTask task;
    @Test
    public void testSyncTasks() throws Exception {
        task.doTaskOne();
        task.doTaskTwo();
        task.doTaskThree();
    }
}

執(zhí)行單元測試,可以看到類似如下輸出:

開始做任務(wù)一
完成任務(wù)一,耗時:6720毫秒
開始做任務(wù)二
完成任務(wù)二,耗時:6604毫秒
開始做任務(wù)三
完成任務(wù)三,耗時:9448毫秒

在這里插入圖片描述

任務(wù)一、任務(wù)二、任務(wù)三順序的執(zhí)行完了,換言之 doTaskOne(),doTaskTwo()doTaskThree() 三個方法按調(diào)用順序的先后執(zhí)行完成。

3. 異步調(diào)用

上述的 同步調(diào)用 雖然順利的執(zhí)行完了三個任務(wù),但是可以看到 執(zhí)行時間比較長,若這三個任務(wù)本身之間 不存在依賴關(guān)系,可以 并發(fā)執(zhí)行 的話,同步調(diào)用在 執(zhí)行效率 方面就比較差,可以考慮通過 異步調(diào)用 的方式來 并發(fā)執(zhí)行。

  • 在Application啟動類上面加上@EnableAsync
  • 創(chuàng)建 AsyncTask類,分別在方法上配置 @Async 注解,將原來的 同步方法 變?yōu)?異步方法。
@Component
public class AsyncTask extends AbstractTask {
    @Async
    public void doTaskOne() throws Exception {
        super.doTaskOne();
    }
    @Async
    public void doTaskTwo() throws Exception {
        super.doTaskTwo();
    }
    @Async
    public void doTaskThree() throws Exception {
        super.doTaskThree();
    }
}

單元測試 用例中,注入 AsyncTask 對象,并在測試用例中執(zhí)行 doTaskOne(),doTaskTwo(),doTaskThree() 三個方法。

@Autowired
private AsyncTask asyncTask;
@Test
public void testAsyncTasks() throws Exception {
    asyncTask.doTaskOne();
    asyncTask.doTaskTwo();
    asyncTask.doTaskThree();
}

執(zhí)行單元測試,可以看到類似如下輸出:

開始做任務(wù)三
開始做任務(wù)一
開始做任務(wù)二

如果反復(fù)執(zhí)行單元測試,可能會遇到各種不同的結(jié)果,比如:

  • 沒有任何任務(wù)相關(guān)的輸出
  • 有部分任務(wù)相關(guān)的輸出
  • 亂序的任務(wù)相關(guān)的輸出

在這里插入圖片描述

原因是目前 doTaskOne(),doTaskTwo()doTaskThree() 這三個方法已經(jīng) 異步并發(fā)執(zhí)行 了。主程序在 異步調(diào)用 之后,主程序并不會理會這三個函數(shù)是否執(zhí)行完成了,由于沒有其他需要執(zhí)行的內(nèi)容,所以程序就 自動結(jié)束 了,導(dǎo)致了任務(wù) 不完整 或是 沒有輸出 相關(guān)內(nèi)容的情況。

注意:@Async所修飾的函數(shù)不要定義為static類型,這樣異步調(diào)用不會生效。

4. 異步回調(diào)

為了讓 doTaskOne(),doTaskTwo(),doTaskThree() 能正常結(jié)束,假設(shè)我們需要統(tǒng)計一下三個任務(wù) 并發(fā)執(zhí)行 共耗時多少,這就需要等到上述三個函數(shù)都完成動用之后記錄時間,并計算結(jié)果。

那么我們?nèi)绾闻袛嗌鲜鋈齻€ 異步調(diào)用 是否已經(jīng)執(zhí)行完成呢?我們需要使用 Future<T> 來返回 異步調(diào)用結(jié)果

  • 創(chuàng)建 AsyncCallBackTask 類,聲明 doTaskOneCallback(),doTaskTwoCallback()doTaskThreeCallback() 三個方法,對原有的三個方法進行包裝。
@Component
public class AsyncCallBackTask extends AbstractTask {
    @Async
    public Future<String> doTaskOneCallback() throws Exception {
        super.doTaskOne();
        return new AsyncResult<>("任務(wù)一完成");
    }
    @Async
    public Future<String> doTaskTwoCallback() throws Exception {
        super.doTaskTwo();
        return new AsyncResult<>("任務(wù)二完成");
    }
    @Async
    public Future<String> doTaskThreeCallback() throws Exception {
        super.doTaskThree();
        return new AsyncResult<>("任務(wù)三完成");
    }
}

單元測試 用例中,注入 AsyncCallBackTask 對象,并在測試用例中執(zhí)行 doTaskOneCallback(),doTaskTwoCallback()doTaskThreeCallback() 三個方法。循環(huán)調(diào)用 FutureisDone() 方法等待三個 并發(fā)任務(wù) 執(zhí)行完成,記錄最終執(zhí)行時間。

@Autowired
private AsyncCallBackTask asyncCallBackTask;
@Test
public void testAsyncCallbackTask() throws Exception {
    long start = currentTimeMillis();
    Future<String> task1 = asyncCallBackTask.doTaskOneCallback();
    Future<String> task2 = asyncCallBackTask.doTaskTwoCallback();
    Future<String> task3 = asyncCallBackTask.doTaskThreeCallback();
    // 三個任務(wù)都調(diào)用完成,退出循環(huán)等待
    while (!task1.isDone() || !task2.isDone() || !task3.isDone()) {
        sleep(1000);
    }
    long end = currentTimeMillis();
    System.out.println("任務(wù)全部完成,總耗時:" + (end - start) + "毫秒");
}

看看都做了哪些改變:

  • 在測試用例一開始記錄開始時間;
  • 在調(diào)用三個異步函數(shù)的時候,返回Future類型的結(jié)果對象;
  • 在調(diào)用完三個異步函數(shù)之后,開啟一個循環(huán),根據(jù)返回的Future對象來判斷三個異步函數(shù)是否都結(jié)束了。若都結(jié)束,就結(jié)束循環(huán);若沒有都結(jié)束,就等1秒后再判斷。
  • 跳出循環(huán)之后,根據(jù)結(jié)束時間 - 開始時間,計算出三個任務(wù)并發(fā)執(zhí)行的總耗時。

在這里插入圖片描述

執(zhí)行一下上述的單元測試,可以看到如下結(jié)果:

開始做任務(wù)三
開始做任務(wù)一
開始做任務(wù)二
完成任務(wù)二,耗時:2572毫秒
完成任務(wù)一,耗時:7333毫秒
完成任務(wù)三,耗時:7647毫秒
任務(wù)全部完成,總耗時:8013毫秒

到此這篇關(guān)于springboot通過Async注解實現(xiàn)異步任務(wù)及回調(diào)的文章就介紹到這了,更多相關(guān)springboot Async注解異步任務(wù)內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • 淺談為什么阿里巴巴要禁用Executors創(chuàng)建線程池

    淺談為什么阿里巴巴要禁用Executors創(chuàng)建線程池

    這篇文章主要介紹了淺談為什么阿里巴巴要禁用Executors創(chuàng)建線程池,文中通過示例代碼介紹的非常詳細,對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2021-02-02
  • Java內(nèi)存模型知識匯總

    Java內(nèi)存模型知識匯總

    本文中,有很多定義和說法,都是筆者自己理解后定義出來的。希望能夠讓讀者可以對Java內(nèi)存模型有更加清晰的認(rèn)識。當(dāng)然,如有偏頗,歡迎指正。
    2018-09-09
  • 高斯混合模型與EM算法圖文詳解

    高斯混合模型與EM算法圖文詳解

    高斯模型就是用高斯概率密度函數(shù)(正態(tài)分布曲線)精確地量化事物,將一個事物分解為若干的基于高斯概率密度函數(shù)(正態(tài)分布曲線)形成的模型
    2021-08-08
  • RestTemplate的URL請求示例

    RestTemplate的URL請求示例

    這篇文章主要為大家介紹了RestTemplate的URL請求示例,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪
    2022-06-06
  • Java 中解決Unsupported major.minor version 51.0的問題

    Java 中解決Unsupported major.minor version 51.0的問題

    本文主要介紹解決Unsupported major.minor version 51.0的問題, 這里給大家整理了詳細資料,有需要的小伙伴可以參考下
    2016-08-08
  • MybatisPlus:使用SQL保留字(關(guān)鍵字)的操作

    MybatisPlus:使用SQL保留字(關(guān)鍵字)的操作

    這篇文章主要介紹了MybatisPlus:使用SQL保留字(關(guān)鍵字)的操作,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧
    2020-11-11
  • springboot結(jié)合ehcache防止惡意刷新請求的實現(xiàn)

    springboot結(jié)合ehcache防止惡意刷新請求的實現(xiàn)

    這篇文章主要介紹了springboot結(jié)合ehcache防止惡意刷新請求的實現(xiàn),文中通過示例代碼介紹的非常詳細,對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2020-12-12
  • Java中JS引擎實現(xiàn)的一句話木馬

    Java中JS引擎實現(xiàn)的一句話木馬

    這篇文章主要為大家介紹了如何利用Java中JS引擎實現(xiàn)的一句話木馬,有需要的朋友可以借鑒參考下,希望能夠有所幫助祝大家多多進步
    2022-02-02
  • SpringBoot Knife4j在線API文檔框架基本使用

    SpringBoot Knife4j在線API文檔框架基本使用

    knife4j是為Java MVC框架集成Swagger生成Api文檔的增強解決方案,這篇文章主要介紹了SpringBoot中使用Knife4J在線API文檔框架,需要的朋友可以參考下
    2022-12-12
  • MyBatis 配置之集合的嵌套方式

    MyBatis 配置之集合的嵌套方式

    這篇文章主要介紹了MyBatis 配置之集合的嵌套方式,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教
    2021-09-09

最新評論