一区二区三区在线-一区二区三区亚洲视频-一区二区三区亚洲-一区二区三区午夜-一区二区三区四区在线视频-一区二区三区四区在线免费观看

服務器之家:專注于服務器技術及軟件下載分享
分類導航

PHP教程|ASP.NET教程|Java教程|ASP教程|編程技術|正則表達式|C/C++|IOS|C#|Swift|Android|VB|R語言|JavaScript|易語言|vb.net|

服務器之家 - 編程語言 - Java教程 - Spring Boot Async異步執行任務過程詳解

Spring Boot Async異步執行任務過程詳解

2020-08-14 00:04猿天地 Java教程

這篇文章主要介紹了Spring Boot Async異步執行任務過程詳解,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友可以參考下

異步調用就是不用等待結果的返回就執行后面的邏輯,同步調用則需要等帶結果再執行后面的邏輯。

通常我們使用異步操作都會去創建一個線程執行一段邏輯,然后把這個線程丟到線程池中去執行,代碼如下:

?
1
2
3
4
5
6
7
8
9
ExecutorService executorService = Executors.newFixedThreadPool(10);
executorService.execute(() -> {
  try {
    // 業務邏輯
  } catch (Exception e) {
    e.printStackTrace();
  } finally {
  }
 });

這樣的方式看起來沒那么優雅,盡管用了java的lambda。在Spring Boot中有一種更簡單的方式來執行異步操作,只需要一個@Async注解即可。

?
1
2
3
4
@Async
public void saveLog() {
  System.err.println(Thread.currentThread().getName());
}

我們可以直接在Controller中調用這個業務方法,它就是異步執行的,會在默認的線程池中去執行。需要注意的是一定要在外部的類中去調用這個方法,如果在本類調用是不起作用的,比如this.saveLog()。 最后在啟動類上開啟異步任務的執行,添加@EnableAsync即可。

另外關于執行異步任務的線程池我們也可以自定義,首先我們定義一個線程池的配置類,用來配置一些參數,具體代碼如下:

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
import org.springframework.boot.context.properties.ConfigurationProperties;
import org.springframework.context.annotation.Configuration;
/**
 * 異步任務線程池配置
 *
 * @author yinjihuan
 */
@Configuration
@ConfigurationProperties(prefix = "spring.task.pool")
public class TaskThreadPoolConfig {
  //核心線程數
  private int corePoolSize = 5;
  //最大線程數
  private int maxPoolSize = 50;
  //線程池維護線程所允許的空閑時間
  private int keepAliveSeconds = 60;
  //隊列長度
  private int queueCapacity = 10000;
  //線程名稱前綴
  private String threadNamePrefix = "FSH-AsyncTask-";
  public String getThreadNamePrefix() {
    return threadNamePrefix;
  }
  public void setThreadNamePrefix(String threadNamePrefix) {
    this.threadNamePrefix = threadNamePrefix;
  }
  public int getCorePoolSize() {
    return corePoolSize;
  }
  public void setCorePoolSize(int corePoolSize) {
    this.corePoolSize = corePoolSize;
  }
  public int getMaxPoolSize() {
    return maxPoolSize;
  }
  public void setMaxPoolSize(int maxPoolSize) {
    this.maxPoolSize = maxPoolSize;
  }
  public int getKeepAliveSeconds() {
    return keepAliveSeconds;
  }
  public void setKeepAliveSeconds(int keepAliveSeconds) {
    this.keepAliveSeconds = keepAliveSeconds;
  }
  public int getQueueCapacity() {
    return queueCapacity;
  }
  public void setQueueCapacity(int queueCapacity) {
    this.queueCapacity = queueCapacity;
  }
}

然后我們重新定義線程池的配置:

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
import java.lang.reflect.Method;
import java.util.concurrent.Executor;
import java.util.concurrent.ThreadPoolExecutor;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.aop.interceptor.AsyncUncaughtExceptionHandler;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.context.annotation.Configuration;
import org.springframework.scheduling.annotation.AsyncConfigurer;
import org.springframework.scheduling.concurrent.ThreadPoolTaskExecutor;
 
@Configuration
public class AsyncTaskExecutePool implements AsyncConfigurer { 
  private Logger logger = LoggerFactory.getLogger(AsyncTaskExecutePool.class);
 
  @Autowired
  private TaskThreadPoolConfig config;
 
  @Override
  public Executor getAsyncExecutor() {
    ThreadPoolTaskExecutor executor = new ThreadPoolTaskExecutor();
    executor.setCorePoolSize(config.getCorePoolSize()); 
    executor.setMaxPoolSize(config.getMaxPoolSize()); 
    executor.setQueueCapacity(config.getQueueCapacity()); 
    executor.setKeepAliveSeconds(config.getKeepAliveSeconds()); 
    executor.setThreadNamePrefix(config.getThreadNamePrefix());
    //線程池對拒絕任務(無線程可用)的處理策略,目前只支持AbortPolicy、CallerRunsPolicy
    //AbortPolicy:直接拋出java.util.concurrent.RejectedExecutionException異常 -->
    //CallerRunsPolicy:主線程直接執行該任務,執行完之后嘗試添加下一個任務到線程池中,可以有效降低向線程池內添加任務的速度 -->
    //DiscardOldestPolicy:拋棄舊的任務、暫不支持;會導致被丟棄的任務無法再次被執行 -->
    //DiscardPolicy:拋棄當前任務、暫不支持;會導致被丟棄的任務無法再次被執行 -->
    executor.setRejectedExecutionHandler(new ThreadPoolExecutor.CallerRunsPolicy());
    executor.initialize(); 
    return executor; 
  }
  @Override
  public AsyncUncaughtExceptionHandler getAsyncUncaughtExceptionHandler() {// 異步任務中異常處理
    return new AsyncUncaughtExceptionHandler() {
      @Override
      public void handleUncaughtException(Throwable arg0, Method arg1, Object... arg2) {
        logger.error("=========================="+arg0.getMessage()+"=======================", arg0);
        logger.error("exception method:" + arg1.getName());
      }
    };
  
}

配置完之后我們的異步任務執行的線程池就是我們自定義的了,我們可以通過在屬性文件里面配置線程池的大小等等信息,也可以使用默認的配置:

spring.task.pool.maxPoolSize=100

最后講下線程池配置的拒絕策略,當我們的線程數量高于線程池的處理速度時,任務會被緩存到本地的隊列中,隊列也是有大小的,如果超過了這個大小,我們需要有拒絕的策略,不然就會內存溢出了,目前支持2種拒絕策略:

  • AbortPolicy: 直接拋出java.util.concurrent.RejectedExecutionException異常
  • CallerRunsPolicy: 主線程直接執行該任務,執行完之后嘗試添加下一個任務到線程池中,可以有效降低向線程池內添加任務的速度
  • 建議大家用CallerRunsPolicy策略,因為當隊列中的任務滿了之后,如果直接拋異常,那么這個任務就會被丟棄,如果是CallerRunsPolicy策略會用主線程去執行,就是同步執行,最起碼這樣任務不會丟棄。

以上就是本文的全部內容,希望對大家的學習有所幫助,也希望大家多多支持服務器之家。

原文鏈接:https://blog.51cto.com/14888386/2516391

延伸 · 閱讀

精彩推薦
主站蜘蛛池模板: 午夜欧美福利视频 | 午夜伦伦电影理论片费看 | 嗯啊好爽视频 | 国产成人影院在线观看 | 小sao货水好多真紧h的视频 | 欧美xxxxx九色视频免费观看 | tiny4k欧美极品在线 | 隔壁老王国产在线精品 | 操大肥b | 国内精品 大秀视频 日韩精品 | 2020年国产精品午夜福利在线观看 | 国产精品一级视频 | 亚洲国产精品一在线观看 | 小小水蜜桃3视频在线观看 小鸟酱喷水 | 亚洲swag精品自拍一区 | 纲手被漫画aⅴ | 日本高清免费看 | 亚洲精品精品一区 | 色啪啪888.com| 婷婷日日夜夜 | 亚洲视频在线观看不卡 | 精品综合久久久久久8888 | 桃子视频www | 日本三不卡 | 无人知晓小说姜璟免费阅读 | 风间由美在线播放 | 俄罗斯13一14处出血视频在线 | 亚洲国产精品久久网午夜 | 日本在线视频网址 | 无遮挡h肉动漫高清在线 | 东京道一本热大交乱 | 色老板最新网站视频地址 | 男人的j插入女人的p | 国产精自产拍久久久久久 | 日本天堂视频 | kayden kross喷水| 污斗罗大陆 | 免费一级毛片在线播放 | 国产手机在线观看 | 日韩欧免费一区二区三区 | 被18号每天强行榨干acg |