springboot定時任務線程阻塞踩坑

場景描述

在使用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;
    }
}
發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章