場景描述
在使用Springboot
整合定時任務,發現當某個定時任務執行出現執行時間過長的情況時會阻塞其他定時任務的執行。
問題定位
後續通過翻查Springboot
的文檔以及打印日誌(輸出當前線程信息)得知問題是由於Springboot
默認使用只有1
個線程的單線程池處理定時任務。
問題覆盤
需要注意示例的Springboot
版本爲2.1.3.RELEASE
。
@Component @Log4j2 public class ScheduledTask { @Scheduled(cron = "0/5 * * * * ?") public void task1() throws InterruptedException { log.info("I am task11111111, current thread: {}", Thread.currentThread()); while (true) { //模擬耗時任務,阻塞10s Thread.sleep(10000); break; } } @Scheduled(cron = "0/5 * * * * ?") public void task2() { log.info("I am task22222222, current thread: {}", Thread.currentThread()); } }
2019-04-24 17:11:15.008 INFO 16868 --- [ scheduling-1] com.example.demo.task.ScheduledTask : I am task22222222, current thread: Thread[scheduling-1,5,main] 2019-04-24 17:11:15.009 INFO 16868 --- [ scheduling-1] com.example.demo.task.ScheduledTask : I am task11111111, current thread: Thread[scheduling-1,5,main] 2019-04-24 17:11:25.009 INFO 16868 --- [ scheduling-1] com.example.demo.task.ScheduledTask : I am task22222222, current thread: Thread[scheduling-1,5,main] 2019-04-24 17:11:30.002 INFO 16868 --- [ scheduling-1] com.example.demo.task.ScheduledTask : I am task22222222, current thread: Thread[scheduling-1,5,main] 2019-04-24 17:11:30.003 INFO 16868 --- [ scheduling-1] com.example.demo.task.ScheduledTask : I am task11111111, current thread: Thread[scheduling-1,5,main] 2019-04-24 17:11:40.004 INFO 16868 --- [ scheduling-1] com.example.demo.task.ScheduledTask : I am task22222222, current thread: Thread[scheduling-1,5,main]
由結果可見,task1與task2由同一個線程Thread[scheduling-1,5,main]執行,也即該定時任務默認使用單線程,並且由於task1阻塞了10s,導致本應5s執行一次的定時任務10s才執行一次。
解決方法
由於使用的Springboot
版本爲2.1.3.RELEASE
,所以有兩種方法解決這個問題
使用Springboot配置
在配置文件中可以配置定時任務可用的線程數:
## 配置可用線程數爲10
spring.task.scheduling.pool.size=10
自定義定時任務的線程池
使用自定義的線程池代替默認的線程池
/** * 定時任務配置類 * @author RJH * create at 2019-03-29 */ @Configuration public class ScheduleConfig { /** * 此處方法名爲Bean的名字,方法名無需固定 * 因爲是按TaskScheduler接口自動注入 * @return */ @Bean public TaskScheduler taskScheduler(){ // Spring提供的定時任務線程池類 ThreadPoolTaskScheduler taskScheduler=new ThreadPoolTaskScheduler(); //設定最大可用的線程數目 taskScheduler.setPoolSize(10); return taskScheduler; } }