1. <strong id="7actg"></strong>
    2. <table id="7actg"></table>

    3. <address id="7actg"></address>
      <address id="7actg"></address>
      1. <object id="7actg"><tt id="7actg"></tt></object>

        SpringBoot中使用異步方法優(yōu)化Service邏輯,提高接口響應(yīng)速度

        共 6097字,需瀏覽 13分鐘

         ·

        2021-11-04 19:24

        程序員的成長(zhǎng)之路
        互聯(lián)網(wǎng)/程序員/技術(shù)/資料共享?
        關(guān)注


        閱讀本文大概需要 5 分鐘。

        來自:blog.csdn.net/weixin_43441509/article/details/119855613

        1. 為什么需要異步方法?

        先說結(jié)論: 合理使用異步方法可以讓業(yè)務(wù)接口快到飛起!
        異步方法適用于邏輯與邏輯之間可以相互分割互不影響的業(yè)務(wù)中, 如生成驗(yàn)證碼和發(fā)送驗(yàn)證碼組成的業(yè)務(wù), 其實(shí)無需等到真正發(fā)送成功驗(yàn)證碼才對(duì)客戶端進(jìn)行響應(yīng), 可以讓短信發(fā)送這一耗時(shí)操作轉(zhuǎn)為異步執(zhí)行, 解耦耗時(shí)操作和核心業(yè)務(wù);
        同理還有文章閱讀的業(yè)務(wù)邏輯 = 查詢文章詳情 + 更新文章閱讀量后再響應(yīng)客戶端, 其實(shí)也無需等到閱讀量更新后才響應(yīng)文章詳情給客戶端, 用戶查看文章是主要邏輯, 而文章閱讀量更新是次要邏輯, 況且閱讀量就算更新失敗一點(diǎn)數(shù)據(jù)偏差也不會(huì)影響用戶閱讀因此這兩個(gè)數(shù)據(jù)庫操作之間的一致性是較弱的, 這類都能用異步事件去優(yōu)化.
        所以說: 恰當(dāng)?shù)脑谖覀兊腟ervice中加入異步方法能大大提高接口的響應(yīng)速度, 提升用戶體驗(yàn)!
        同步執(zhí)行(同在一個(gè)線程中):
        異步執(zhí)行(開啟額外線程來執(zhí)行):

        2. SpringBoot中的異步方法支持

        其實(shí), 在SpringBoot中并不需要我們自己去創(chuàng)建維護(hù)線程或者線程池來異步的執(zhí)行方法, SpringBoot已經(jīng)提供了異步方法支持注解.
        1. @EnableAsync?//?使用異步方法時(shí)需要提前開啟(在啟動(dòng)類上或配置類上)

        2. @Async?//?被async注解修飾的方法由SpringBoot默認(rèn)線程池(SimpleAsyncTaskExecutor)執(zhí)行

        比如使用Spring的異步支持實(shí)現(xiàn)文章查詢并增加閱讀量
        Service層:
        1. @Service

        2. public?class?ArticleServiceImpl?{

        3. ????//?查詢文章

        4. ????public?String?selectArticle()?{

        5. ????????//?TODO?模擬文章查詢操作

        6. ????????System.out.println("查詢?nèi)蝿?wù)線程"+Thread.currentThread().getName());

        7. ????????return?"文章詳情";

        8. ????}

        9. ????//?文章閱讀量+1

        10. ????@Async

        11. ????public?void?updateReadCount()?{

        12. ????????//?TODO?模擬耗時(shí)操作

        13. ????????try?{

        14. ????????????Thread.sleep(3000);

        15. ????????}?catch?(InterruptedException?e)?{

        16. ????????????e.printStackTrace();

        17. ????????}

        18. ????????System.out.println("更新任務(wù)線程"+Thread.currentThread().getName());

        19. ????}

        20. }

        Controller層:
        1. @RestController

        2. public?class?AsyncTestController?{

        3. ????@Autowired

        4. ????private?ArticleServiceImpl?articleService;

        5. ????/**

        6. ?????*?模擬獲取文章后閱讀量+1

        7. ?????*/

        8. ????@PostMapping("/article")

        9. ????public?String?getArticle()?{

        10. ????????//?查詢文章

        11. ????????String?article?=?articleService.selectArticle();

        12. ????????//?閱讀量+1

        13. ????????articleService.updateReadCount();

        14. ????????System.out.println("文章閱讀業(yè)務(wù)執(zhí)行完畢");

        15. ????????return?article;

        16. ????}

        17. }

        測(cè)試結(jié)果: 我們可以感受到接口響應(yīng)速度大大提升, 而且從日志中key看到兩個(gè)執(zhí)行任務(wù)是在不同的線程中執(zhí)行的

        3. 自定義線程池執(zhí)行異步方法

        SpringBoot為我們默認(rèn)提供了線程池(SimpleAsyncTaskExecutor)來執(zhí)行我們的異步方法, 我們也可以自定義自己的線程池.
        第一步配置自定義線程池
        1. @EnableAsync?//?開啟多線程,?項(xiàng)目啟動(dòng)時(shí)自動(dòng)創(chuàng)建

        2. @Configuration

        3. public?class?AsyncConfig?{

        4. ????@Bean("customExecutor")

        5. ????public?ThreadPoolTaskExecutor?asyncOperationExecutor()?{

        6. ????????ThreadPoolTaskExecutor?executor?=?new?ThreadPoolTaskExecutor();

        7. ????????//?設(shè)置核心線程數(shù)

        8. ????????executor.setCorePoolSize(8);

        9. ????????//?設(shè)置最大線程數(shù)

        10. ????????executor.setMaxPoolSize(20);

        11. ????????//?設(shè)置隊(duì)列大小

        12. ????????executor.setQueueCapacity(Integer.MAX_VALUE);

        13. ????????//?設(shè)置線程活躍時(shí)間(秒)

        14. ????????executor.setKeepAliveSeconds(60);

        15. ????????//?設(shè)置線程名前綴+分組名稱

        16. ????????executor.setThreadNamePrefix("AsyncOperationThread-");

        17. ????????executor.setThreadGroupName("AsyncOperationGroup");

        18. ????????//?所有任務(wù)結(jié)束后關(guān)閉線程池

        19. ????????executor.setWaitForTasksToCompleteOnShutdown(true);

        20. ????????//?初始化

        21. ????????executor.initialize();

        22. ????????return?executor;

        23. ????}

        24. }

        第二步, 在@Async注解上指定執(zhí)行的線程池即可
        1. //?文章閱讀量+1

        2. @Async("customExecutor")

        3. public?void?updateReadCount()?{

        4. ????//?TODO?模擬耗時(shí)操作

        5. ????try?{

        6. ????????Thread.sleep(3000);

        7. ????}?catch?(InterruptedException?e)?{

        8. ????????e.printStackTrace();

        9. ????}

        10. ????System.out.println("更新文章閱讀量線程"+Thread.currentThread().getName());

        11. }

        5. 如何捕獲(無返回值的)異步方法中的異常

        以實(shí)現(xiàn)AsyncConfigurer接口的getAsyncExecutor方法和getAsyncUncaughtExceptionHandler方法改造配置類
        自定義異常處理類CustomAsyncExceptionHandler
        1. @EnableAsync?//?開啟多線程,?項(xiàng)目啟動(dòng)時(shí)自動(dòng)創(chuàng)建

        2. @Configuration

        3. public?class?AsyncConfig?implements?AsyncConfigurer?{

        4. ????@Override

        5. ????public?Executor?getAsyncExecutor()?{

        6. ????????ThreadPoolTaskExecutor?executor?=?new?ThreadPoolTaskExecutor();

        7. ????????//?設(shè)置核心線程數(shù)

        8. ????????executor.setCorePoolSize(8);

        9. ????????//?設(shè)置最大線程數(shù)

        10. ????????executor.setMaxPoolSize(20);

        11. ????????//?設(shè)置隊(duì)列大小

        12. ????????executor.setQueueCapacity(Integer.MAX_VALUE);

        13. ????????//?設(shè)置線程活躍時(shí)間(秒)

        14. ????????executor.setKeepAliveSeconds(60);

        15. ????????//?設(shè)置線程名前綴+分組名稱

        16. ????????executor.setThreadNamePrefix("AsyncOperationThread-");

        17. ????????executor.setThreadGroupName("AsyncOperationGroup");

        18. ????????//?所有任務(wù)結(jié)束后關(guān)閉線程池

        19. ????????executor.setWaitForTasksToCompleteOnShutdown(true);

        20. ????????//?初始化

        21. ????????executor.initialize();

        22. ????????return?executor;

        23. ????}

        24. ????@Override

        25. ????public?AsyncUncaughtExceptionHandler?getAsyncUncaughtExceptionHandler()?{

        26. ????????return?new?CustomAsyncExceptionHandler();

        27. ????}

        28. }

        1. public?class?CustomAsyncExceptionHandler?implements?AsyncUncaughtExceptionHandler?{

        2. ?

        3. ????@Override

        4. ????public?void?handleUncaughtException(Throwable?throwable,?Method?method,?Object...?obj)?{

        5. ????????System.out.println("異常捕獲---------------------------------");

        6. ????????System.out.println("Exception?message?-?"?+?throwable.getMessage());

        7. ????????System.out.println("Method?name?-?"?+?method.getName());

        8. ????????for?(Object?param?:?obj)?{

        9. ????????????System.out.println("Parameter?value?-?"?+?param);

        10. ????????}

        11. ????????System.out.println("異常捕獲---------------------------------");

        12. ????}

        13. ?????

        14. }

        5. 如何獲取(有返回值)異步方法的返回值

        使用Future類及其子類來接收異步方法返回值
        注意:
        • 無返回值的異步方法拋出異常不會(huì)影響Controller的主要業(yè)務(wù)邏輯

        • 有返回值的異步方法拋出異常會(huì)影響Controller的主要業(yè)務(wù)邏輯

        1. //?異步方法---------------------------------------------------------------------

        2. @Async

        3. ????public?CompletableFuture?updateReadCountHasResult()?{

        4. ????????//?TODO?模擬耗時(shí)操作

        5. ????????try?{

        6. ????????????Thread.sleep(3000);

        7. ????????}?catch?(InterruptedException?e)?{

        8. ????????????e.printStackTrace();

        9. ????????}

        10. ????????System.out.println("更新文章閱讀量線程"+Thread.currentThread().getName());

        11. ????????return?CompletableFuture.completedFuture(100?+?1);

        12. ????}

        13. //?Controller調(diào)用---------------------------------------------------------------------

        14. @GetMapping("/article")

        15. public?String?getArticle()?throws?ExecutionException,?InterruptedException?{

        16. ????//?查詢文章

        17. ????String?article?=?articleService.selectArticle();

        18. ????//?閱讀量+1

        19. ????CompletableFuture?future?=?articleService.updateReadCountHasResult();

        20. ????int?count?=?0;

        21. ????//?循環(huán)等待異步請(qǐng)求結(jié)果

        22. ????while?(true)?{

        23. ????????if(future.isCancelled())?{

        24. ????????????System.out.println("異步任務(wù)取消");

        25. ????????????break;

        26. ????????}

        27. ????????if?(future.isDone())?{

        28. ????????????count?=?future.get();

        29. ????????????System.out.println(count);

        30. ????????????break;

        31. ????????}

        32. ????}

        33. ????System.out.println("文章閱讀業(yè)務(wù)執(zhí)行完畢");

        34. ????return?article?+?count;

        35. }

        6. 異步方法帶來的問題/拓展

        • 異步方法只能聲明在Service方法中在Controller直接調(diào)用才會(huì)生效, 異步方法被同級(jí)Service方法調(diào)用不會(huì)生效, 很奇怪?

        • 異步方法 + 事務(wù)能順利執(zhí)行嗎? 或許事務(wù)操作應(yīng)該和異步操作分離開, 被Controller層調(diào)用時(shí)事務(wù)操作在前, 異步操作在后

        • 異步方法執(zhí)行失敗后對(duì)Controller前半部分的非異步操作無影響, 因此說異步方法在整個(gè)業(yè)務(wù)邏輯中不是100%可靠的, 對(duì)于強(qiáng)一致性的業(yè)務(wù)來說不適用

        • 還是消息中間件更為強(qiáng)大, RabbitMQ, Kafka…

        推薦閱讀:

        火遍全國(guó)的網(wǎng)絡(luò)熱梗“yyds”,創(chuàng)造者被判刑3年

        Hbase與MySQL對(duì)比,區(qū)別是什么?

        朕已閱?

        瀏覽 35
        點(diǎn)贊
        評(píng)論
        收藏
        分享

        手機(jī)掃一掃分享

        分享
        舉報(bào)
        評(píng)論
        圖片
        表情
        推薦
        點(diǎn)贊
        評(píng)論
        收藏
        分享

        手機(jī)掃一掃分享

        分享
        舉報(bào)
        1. <strong id="7actg"></strong>
        2. <table id="7actg"></table>

        3. <address id="7actg"></address>
          <address id="7actg"></address>
          1. <object id="7actg"><tt id="7actg"></tt></object>
            亚洲AV无码成人精品国产 | 97一期涩涩97片久久久久久久 | 黄片入口| 成年人视频免费 | 夜色成人电影 | 射得好电影 | A片在线视频免费观看 | 成人做爱毛片 | 东北骚妇大战黑人视频 | 一区二区三区综合 |