<kbd id="afajh"><form id="afajh"></form></kbd>
<strong id="afajh"><dl id="afajh"></dl></strong>
    <del id="afajh"><form id="afajh"></form></del>
        1. <th id="afajh"><progress id="afajh"></progress></th>
          <b id="afajh"><abbr id="afajh"></abbr></b>
          <th id="afajh"><progress id="afajh"></progress></th>

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

          共 6097字,需瀏覽 13分鐘

           ·

          2021-11-04 19:24

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


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

          來(lái)自:blog.csdn.net/weixin_43441509/article/details/119855613

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

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

          2. SpringBoot中的異步方法支持

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

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

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

          2. public?class?ArticleServiceImpl?{

          3. ????//?查詢(xún)文章

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

          5. ????????//?TODO?模擬文章查詢(xún)操作

          6. ????????System.out.println("查詢(xún)?nèi)蝿?wù)線(xiàn)程"+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ù)線(xiàn)程"+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. ????????//?查詢(xún)文章

          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ù)是在不同的線(xiàn)程中執(zhí)行的

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

          SpringBoot為我們默認(rèn)提供了線(xiàn)程池(SimpleAsyncTaskExecutor)來(lái)執(zhí)行我們的異步方法, 我們也可以自定義自己的線(xiàn)程池.
          第一步配置自定義線(xiàn)程池
          1. @EnableAsync?//?開(kāi)啟多線(xiàn)程,?項(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è)置核心線(xiàn)程數(shù)

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

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

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

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

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

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

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

          15. ????????//?設(shè)置線(xiàn)程名前綴+分組名稱(chēng)

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

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

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

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

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

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

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

          23. ????}

          24. }

          第二步, 在@Async注解上指定執(zhí)行的線(xiàn)程池即可
          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("更新文章閱讀量線(xiàn)程"+Thread.currentThread().getName());

          11. }

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

          以實(shí)現(xiàn)AsyncConfigurer接口的getAsyncExecutor方法和getAsyncUncaughtExceptionHandler方法改造配置類(lèi)
          自定義異常處理類(lèi)CustomAsyncExceptionHandler
          1. @EnableAsync?//?開(kāi)啟多線(xiàn)程,?項(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è)置核心線(xiàn)程數(shù)

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

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

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

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

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

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

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

          15. ????????//?設(shè)置線(xiàn)程名前綴+分組名稱(chēng)

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

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

          18. ????????//?所有任務(wù)結(jié)束后關(guān)閉線(xià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類(lèi)及其子類(lèi)來(lái)接收異步方法返回值
          注意:
          • 無(wú)返回值的異步方法拋出異常不會(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("更新文章閱讀量線(xiàn)程"+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. ????//?查詢(xún)文章

          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. 異步方法帶來(lái)的問(wèn)題/拓展

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

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

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

          • 還是消息中間件更為強(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)
          <kbd id="afajh"><form id="afajh"></form></kbd>
          <strong id="afajh"><dl id="afajh"></dl></strong>
            <del id="afajh"><form id="afajh"></form></del>
                1. <th id="afajh"><progress id="afajh"></progress></th>
                  <b id="afajh"><abbr id="afajh"></abbr></b>
                  <th id="afajh"><progress id="afajh"></progress></th>
                  成人操骚逼逼 | 国际精品久久久 | 18成人网站在线看 | 免费看欧美一级片 | 色无五月 |