最近在做导出,用到了异步,使用了@Async的注解,springboot 会默认线程池,不过一般我们会自定义线程池
配置@Async
- springBoot启动类的配置
@SpringBootApplication
@EnableAsync
public class MyApplication {
public static void main(String[] args) {
SpringApplication.run(MyApplication.class, args);
}
}
- Java代码结合@Configuration注解的配置方式
@Configuration
public class ThreadPoolConfig {
/**
* 核心线程数,基于当前线程池是 磁盘IO还是网络IO,磁盘IO=CPU核数,网络IO=CPU核数*2, 目前Business异步线程池都是用于网络异步通知消息
*/
private static final int CORE_POOL_SIZE = Runtime.getRuntime().availableProcessors() * 2;
/**
* 最大线程数,
*/
private static final int MAX_POOL_SIZE = 256;
/**
* 线程空闲最大时间
*/
private static final int KEEP_ALIVE_TIME = 60;
/**
* 用来设置线程池关闭的时候等待所有任务都完成再继续销毁其他的Bean
*/
private static final Boolean WAIT_FOR_TASKS_TO_COMPLETE_ON_SHUTDOWN = true;
/**
* 队列容量
*/
private static final int QUEUE_CAPACITY = 2048;
/**
* 线程池前缀
*/
private static final String THREAD_NAME_PREFIX = "planet-bops-executor-";
@Bean(name = { "bopsAsyncExecutor" })
public Executor bopsAsyncExecutor() {
ThreadPoolTaskExecutor taskExecutor = new ThreadPoolTaskExecutor();
taskExecutor.setThreadNamePrefix(THREAD_NAME_PREFIX);
taskExecutor.setCorePoolSize(CORE_POOL_SIZE);
taskExecutor.setMaxPoolSize(MAX_POOL_SIZE);
taskExecutor.setKeepAliveSeconds(KEEP_ALIVE_TIME);
taskExecutor.setQueueCapacity(QUEUE_CAPACITY);
taskExecutor.setWaitForTasksToCompleteOnShutdown(WAIT_FOR_TASKS_TO_COMPLETE_ON_SHUTDOWN);
taskExecutor.setRejectedExecutionHandler(new ThreadPoolExecutor.CallerRunsPolicy());
taskExecutor.initialize();
return TtlExecutors.getTtlExecutor(taskExecutor);
}
- 使用@Async注意事项:如下方式会失效
- 异步方法使用static修饰
- 异步类没有使用@Component注解(或其他注解)导致spring无法扫描到异步类
- 异步方法不能与被调用的异步方法在同一个类中
- 类中需要使用@Autowired或@Resource等注解自动注入,不能自己手动new对象
- 如果使用SpringBoot框架必须在启动类中增加@EnableAsync注解