成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

聊聊 Spring 異步任務教程

開發 前端
阿粉最近碰到一個場景,用戶注冊之后需要發送郵件給其郵箱。原先設計中,這是一個同步過程,注冊方法需要等待郵件發送成功才能返回。

 阿粉最近碰到一個場景,用戶注冊之后需要發送郵件給其郵箱。原先設計中,這是一個同步過程,注冊方法需要等待郵件發送成功才能返回。

[[331996]]

由于郵件發送流程對于注冊來說并不是一個關鍵節點,我們可以將郵件發送異步執行,減少注冊方法執行時間。

我們可以自己創建線程池,然后執行異步任務,示例代碼如下:

  1. // 生產使用線程池的最佳實踐,一定要自定義線程池,不要嫌麻煩,使用 Executors 創建線程池 
  2. private ThreadPoolExecutor threadPool = 
  3.         new ThreadPoolExecutor(5, 
  4.                 10, 
  5.                 60l, 
  6.                 TimeUnit.SECONDS, 
  7.                 new LinkedBlockingDeque<>(200), 
  8.                 new ThreadFactoryBuilder().setNameFormat("register-%d").build()); 
  9.  
  10. /** 
  11.  * 使用線程池執行發送郵件的任務 
  12.  */ 
  13. private void sendEmailByThreadPool() { 
  14.     threadPool.submit(() -> emailService.sendEmail()); 

ps: 生產使用線程池的最佳實踐,一定要自定義線程池,根據業務場景設置合理的線程池參數,另外給線程設置具有明確意義的前綴,這樣排查問題就非常簡單。

千萬不要為了方便,使用 Executors 相關方法創建線程池。

上面代碼中使用線程池完成了發送郵件的異步任務,可以看到這個示例還是有點麻煩,我們不僅要自定義線程池,還需要在創建相關任務執行類。

Spring 提供執行異步任務功能,我們使用一個注解就可以輕松完成上面的功能。

今天阿粉就來講解一下如何使用 Spring 異步任務,以及 Spring 異步任務使用過程中一些注意點。

異步任務使用方式

Spring 異步任務需要在相關的方法上設置 @Async 注解,這里為了舉例,我們創建一個 EmailService 類,專用完成郵件服務。

代碼如下所示:

  1. @Slf4j 
  2. @Service 
  3. public class EmailService { 
  4.  
  5.     /** 
  6.      * 異步發送任務 
  7.      * 
  8.      * @throws InterruptedException 
  9.      */ 
  10.     @SneakyThrows 
  11.     @Async 
  12.     public void sendEmailAsync() { 
  13.         log.info("使用 Spring 異步任務發送郵件示例"); 
  14.         // 模擬郵件發送耗時 
  15.         TimeUnit.SECONDS.sleep(2l); 
  16.     } 

這里要注意了,Spring 異步任務默認關閉的,我們需要使用 @EnableAsync開啟異步任務。

如果還在使用 Spring XML 配置,我們需要配置如下配置:

  1. <task:annotation-driven/> 

上述配置完成之后,我們只需要在調用方,比如上一層 Controller 注入這個 EmailService ,然后直接調用這個方法,該方法將會在異步線程中執行。

  1. @Slf4j 
  2. @RestController 
  3. public class RegisterController { 
  4.  
  5.     @Autowired 
  6.     EmailService emailService; 
  7.  
  8.     @RequestMapping("register"
  9.     public String register() { 
  10.      log.info("注冊流程開始"); 
  11.      emailService.sendEmailAsync(); 
  12.         return "success"
  13.     } 
  14.  } 

輸出日志如下:

從日志上可以看到,兩個方法執行線程不一樣,這就說明了EmailService#sendEmailAsync 被異步線程成功執行。

帶有返回值的異步任務

上面的異步任務比較簡單,但是有時我們有需要獲取異步任務返回值。

如果使用線程池執行異步任務,我們可以使用 threadPool#submit 獲取返回對象 Future,接著我們就可以調用其內 get 方法,獲取返回結果。

在 Spring 異步任務中,我們也可以使用 Future 獲取返回結果,示例代碼如下:

  1. @Async 
  2. @SneakyThrows 
  3. public Future<String> sendEmailAsyncWithResult() { 
  4.     log.info("使用 Spring 異步任務發送郵件,并且獲取任務返回結果示例"); 
  5.     TimeUnit.SECONDS.sleep(2l); 
  6.     return AsyncResult.forValue("success"); 

這里需要注意,這里返回對象我們需要使用 Spring 內部類 AsyncResult。

Controller 層調用代碼如下所示:

  1.  private void sendEmailWithResult() { 
  2.         Future<String> future = emailService.sendEmailAsyncWithResult(); 
  3.         try { 
  4.             String result = future.get(); 
  5.         } catch (InterruptedException e) { 
  6.             e.printStackTrace(); 
  7.         } catch (ExecutionException e) { 
  8.             e.printStackTrace(); 
  9.         } 
  10.  
  11.     } 

我們知道 Future#get 方法將會一直阻塞,直到異步任務執行成功。

有時候我們獲取異步任務的返回值是為了做一下后續業務,但是主流程方法是無需返回異步任務的返回值。如果我們使用了 Future#get方法,主流程就會一直被阻塞。

對于這種場景,我們可以使用 org.springframework.util.concurrent.ListenableFuture稍微改造一下上面的方法。

ListenableFuture 這個類允許我們注冊回調函數,一旦異步任務執行成功,或者執行異常,將會立刻執行回調函數。通過這種方式就可以不用阻塞執行的主線程。

示例代碼如下:

  1. @Async 
  2. @SneakyThrows 
  3. public ListenableFuture<String> sendEmailAsyncWithListenableFuture() { 
  4.     log.info("使用 Spring 異步任務發送郵件,并且獲取任務返回結果示例"); 
  5.     TimeUnit.SECONDS.sleep(2l); 
  6.     return AsyncResult.forValue("success"); 

Controller 層代碼如下所示:

  1. ListenableFuture<String> listenableFuture = emailService.sendEmailAsyncWithListenableFuture(); 
  2. // 異步回調處理 
  3. listenableFuture.addCallback(new SuccessCallback<String>() { 
  4.     @Override 
  5.     public void onSuccess(String result) { 
  6.         log.info("異步回調處理返回值"); 
  7.  
  8.     } 
  9. }, new FailureCallback() { 
  10.     @Override 
  11.     public void onFailure(Throwable ex) { 
  12.         log.error("異步回調處理異常",ex); 
  13.     } 
  14. }); 

看到這里,如果有同學有疑惑,我們返回對象是 AsyncResult,為什么方法返回類可以是 Future,又可以是 ListenableFuture?

看完這張類繼承關系,大家應該就知道答案了。

異常處理方式

異步任務中異常處理方式,不是很難,我們只要在方法中將整個代碼塊 try...catch 即可。

  1. try { 
  2.  // 其他代碼 
  3. } catch (Exception e) { 
  4.     e.printStackTrace(); 

一般來說,我們只需要捕獲 Exception 異常,就可以應對大部分情況

但是極端情況下,比如方法內發生 OOM,將會拋出 OutOfMemoryError。如果發生Error 錯誤,以上的捕獲代碼就會失效。

Spring 的異步任務,默認提供幾種異常處理方式,可以統一處理異步任務中的發生的異常。

帶有返回值的異常處理方式

如果我們使用帶有返回值的異步任務,處理方式就比較簡單了,我們只需要捕獲 Future#get 拋出的異常就好了。

  1. Future<String> future = emailService.sendEmailAsyncWithResult(); 
  2. try { 
  3.     String result = future.get(); 
  4. } catch (InterruptedException e) { 
  5.     e.printStackTrace(); 
  6. } catch (ExecutionException e) { 
  7.     e.printStackTrace(); 

如果我們使用 ListenableFuture 注冊回調函數處理,那我們在方法內增加一個 FailureCallback,在這個實現類處理相關異常即可。

  1. ListenableFuture<String> listenableFuture = emailService.sendEmailAsyncWithListenableFuture(); 
  2. // 異步回調處理 
  3. listenableFuture.addCallback(new SuccessCallback<String>() { 
  4.     @Override 
  5.     public void onSuccess(String result) { 
  6.         log.info("異步回調處理返回值"); 
  7.  
  8.     } 
  9.     // 異常處理 
  10. }, new FailureCallback() { 
  11.     @Override 
  12.     public void onFailure(Throwable ex) { 
  13.         log.error("異步回調處理異常",ex); 
  14.     } 
  15. }); 

統一異常處理方式

沒有返回值的異步任務處理方式就比較復雜了,我們需要繼承 AsyncConfigurerSupport,實現 getAsyncUncaughtExceptionHandler 方法,示例代碼如下:

  1. @Slf4j 
  2. @Configuration 
  3. public class AsyncErrorHandler extends AsyncConfigurerSupport { 
  4.  
  5.     @Override 
  6.     public AsyncUncaughtExceptionHandler getAsyncUncaughtExceptionHandler() { 
  7.         AsyncUncaughtExceptionHandler handler = (throwable, method, objects) -> { 
  8.             log.error("全局異常捕獲", throwable); 
  9.         }; 
  10.         return handler; 
  11.     } 
  12.  

ps:這個異常處理方式只能處理未帶返回值的異步任務。

異步任務使用注意點

異步線程池設置

Spring 異步任務默認使用 Spring 內部線程池 SimpleAsyncTaskExecutor 。

這個線程池比較坑爹,不會復用線程。也就是說來一個請求,將會新建一個線程。極端情況下,如果調用次數過多,將會創建大量線程。

Java 中的線程是會占用一定的內存空間 ,所以創建大量的線程將會導致 OOM 錯誤。

所以如果需要使用異步任務,我們需要一定要使用自定義線程池替換默認線程池。

XML 配置方式

如果當前使用 Spring XML 配置方式,我們可以使用如下配置設置線程池:

  1. <task:annotation-driven/> 
  2. <task:executor id="executor" pool-size="10" queue-capacity="200"/> 

注解方式

如果注解方式配置,配置方式如下:

  1. @Configuration 
  2. public class AsyncConfiguration { 
  3.  
  4.  
  5.     @Bean 
  6.     public ThreadPoolTaskExecutor taskExecutor() { 
  7.         ThreadPoolTaskExecutor executor = new ThreadPoolTaskExecutor(); 
  8.         executor.setThreadNamePrefix("task-Executor-"); 
  9.         executor.setMaxPoolSize(10); 
  10.         executor.setCorePoolSize(5); 
  11.         executor.setQueueCapacity(200); 
  12.         // 還有其他參數可以設置 
  13.         return executor; 
  14.     } 

只要我們配置了這個線程池Bean,Spring 的異步任務都將會使用該線程池執行。

如果我們應用配置了多個線程池Bean,異步任務需要指定使用某個線程池執行,我們只需要在 @Async注解上設置相應 Bean 的名字即可。示例代碼如下:

  1. @Async("taskExecutor"
  2. public void sendEmailAsync() { 
  3.     log.info("使用 Spring 異步任務發送郵件示例"); 
  4.     TimeUnit.SECONDS.sleep(2l); 

Spring Boot 方式

如果是 SpringBoot 項目,從阿粉的測試情況來看,默認將會創建核心線程數為 8,最大線程數為 Integer.MAX_VALUE,隊列數也為 Integer.MAX_VALUE線程池。

雖然上面的線程池不用擔心創建過多線程的問題,不是還是有可能隊列任務過多,導致 OOM 的問題。所以還是建議使用自定義線程池嗎,或者在配置文件修改默認配置,例如:

  1. spring.task.execution.pool.core-size=10 
  2. spring.task.execution.pool.max-size=20 
  3. spring.task.execution.pool.queue-capacity=200 

ps:如果我們使用注解方式自定義了一個線程池,那么 Spring 異步任務都將會使用這個線程池。通過 SpringBoot 配置文件創建的線程池將會失效。

異步方法失效

Spring 異步任務背后原理是使用 AOP ,而使用 Spring AOP 時我們需要注意,切勿在方法內部調用其他使用 AOP 的方法,可能有點拗口,我們來看下代碼:

  1. @Async 
  2. @SneakyThrows 
  3. public ListenableFuture<String> sendEmailAsyncWithListenableFuture() { 
  4.     // 這樣調用,sendEmailAsync 不會異步執行 
  5.     sendEmailAsync(); 
  6.     log.info("使用 Spring 異步任務發送郵件,并且獲取任務返回結果示例"); 
  7.     TimeUnit.SECONDS.sleep(2l); 
  8.     return AsyncResult.forValue("success"); 
  9.  
  10. /** 
  11.      * 異步發送任務 
  12.      * 
  13.      * @throws InterruptedException 
  14.      */ 
  15. @SneakyThrows 
  16. @Async("taskExecutor"
  17. public void sendEmailAsync() { 
  18.     log.info("使用 Spring 異步任務發送郵件示例"); 
  19.     TimeUnit.SECONDS.sleep(2l); 

上面兩個方法都處于同一個類中,這樣調用將會導致 AOP 失效,無法起到 AOP 的效果。

其他類似的 @Transactional,以及自定義的 AOP 注解都會有這個問題,大家使用過程,千萬需要注意這一點。

總結

Spring 異步任務幫我們大大解決簡化開發了流程,只要使用一個@Async就可以輕松解決異步任務。

不過,雖然使用方式比較簡單,大家使用過程一定要注意設置合理的線程池。

 

責任編輯:武曉燕 來源: Java極客技術
相關推薦

2023-07-31 08:05:30

Spring任務調度

2021-06-02 09:01:19

JavaScript 前端異步編程

2024-01-31 08:41:43

異步設計項目

2021-06-28 14:13:35

Jenkins服務器程序

2023-04-28 08:43:46

2021-06-04 08:48:46

Spring ClouMaven Centr版本

2024-10-23 08:13:30

Spring響應式編程

2024-05-23 11:26:02

2024-04-02 08:27:19

異步任務抽象

2016-11-28 09:08:43

java系統異步非阻塞

2022-07-01 08:00:44

異步編程FutureTask

2021-03-08 00:11:02

Spring注解開發

2021-10-28 19:28:04

數據庫開發Spring

2022-06-02 10:18:24

函數計算異步

2023-01-03 10:38:04

函數計算技術

2024-02-19 00:00:00

分布式定時任務框架

2014-04-24 09:49:57

Android測試異步任務

2024-10-14 13:12:59

2023-11-03 14:32:38

2018-06-20 15:33:44

Spring BootJava 9JDK
點贊
收藏

51CTO技術棧公眾號

主站蜘蛛池模板: 欧美一级特黄aaa大片在线观看 | 亚洲网站在线观看 | 精品久久成人 | 久久av一区 | 一区二区三区视频在线免费观看 | 亚洲精品乱码久久久久久按摩观 | 性色的免费视频 | 成人免费观看视频 | 国产精品久久久久久久久久久新郎 | 天堂av免费观看 | 欧美精品在线免费 | 欧美成人精品一区二区三区 | 亚洲国产精品99久久久久久久久 | 偷拍自拍在线观看 | 久久久国产精品 | 日韩欧美在 | 精品在线 | 欧美色性 | 久久精品美女 | 久久久久久国产精品久久 | 欧美精品一区在线 | 欧美综合久久 | 国产精品欧美一区二区 | 欧美999| 日韩午夜场 | 精品国产乱码久久久久久a丨 | 日韩精品无码一区二区三区 | 91看片官网| 男人天堂99| 视频一区欧美 | 91一区二区在线观看 | 日韩在线视频免费观看 | 国产精品一码二码三码在线 | 日本黄色不卡视频 | 亚洲女人的天堂 | 涩涩视频网站在线观看 | 精品国产一区二区 | 欧美福利精品 | 久久国产一区二区三区 | 91久久爽久久爽爽久久片 | 日韩精品一区二区三区视频播放 |