为什么需要多线程定时任务:
默认情况下,Spring Boot定时任务是按单线程方式执行的,也就是说,如果同一时刻有两个定时任务需要执行,那么只能在一个定时任务完成之后再执行下一个。如果只有一个定时任务,这样做肯定没问题;当定时任务增多时,如果一个任务被阻塞,则会导致其他任务无法正常执行。要解决这个问题,需要配置任务调度线程池。
如何实现
1.增加多线程配置类
package com.yangjunbo.JPADemo.config;
import org.springframework.context.annotation.Bean;
import org.springframework.scheduling.concurrent.ThreadPoolTaskExecutor;
import java.util.concurrent.Executor;
public class SchedulerConfig {
@Bean
public Executor taskScheduler() {
ThreadPoolTaskExecutor executor = new ThreadPoolTaskExecutor();
executor.setCorePoolSize(3);
executor.setMaxPoolSize(10);
executor.setQueueCapacity(3);
executor.initialize();
return executor;
}
}
设置执行线程池为3,最大线程数为10。
2.创建schedulertask定时任务类
package com.yangjunbo.JPADemo.scheduled;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.scheduling.annotation.Async;
import org.springframework.scheduling.annotation.EnableAsync;
import org.springframework.scheduling.annotation.EnableScheduling;
import org.springframework.scheduling.annotation.Scheduled;
import org.springframework.stereotype.Component;
import java.text.SimpleDateFormat;
import java.util.Date;
@EnableAsync //开启异步事件支持
@Component
public class SchedulerTask {
private static final Logger logger = LoggerFactory.getLogger(SchedulerTask.class);
@Async
@Scheduled(cron="*/10 * * * * ?")
public void taskCron(){
SimpleDateFormat dateFormat = new SimpleDateFormat("HH:mm:ss");
logger.info("现在时间Scheduled1: " + dateFormat.format(new Date()));
}
@Async
@Scheduled(fixedRate = 5000)
public void taskFixed() {
SimpleDateFormat dateFormat = new SimpleDateFormat("HH:mm:ss");
logger.info("现在时间Scheduled2: " + dateFormat.format(new Date()));
}
}
3.启动程序测试
在上面的示例中,定时任务类SechedulerTask增加了@EnableAsync注解,开启了异步事件支持。同时,在定时方法上增加@Async注解,使任务能够异步执行,这样各个后台任务就不会阻塞。
通过后台日志可以看到,Spring Boot启动线程池负责调度执行后台任务,各个后台任务之间相对独立、互不影响。
参考书籍《springboot从入门到实战-章为忠著》