6e64cac26fdab755f661a04a33ce9973.png

前言

SpringBoot 开启一个定时任务非常简单,在方法上加上 @Scheduled 注解跟配合 @EnableScheduling 注解开启就能够开启一个定时任务。

在使用 Springboot 整合定时任务,发现当某个定时任务执行出现执行时间过长的情况时会阻塞其他定时任务的执行。

问题定位

问题是由于 Springboot 默认使用只有 1 个线程的单线程池处理定时任务。

这样对于我们的多任务调度可能会是致命的,当多个任务并发(或需要在同一时间)执行时,任务调度器就会出现时间漂移,任务执行时间将不确定。

问题复位

这里所使用的 Springboot 版本为 2.3.5.RELEASE 。

第一步:在启动类上新增 @EnableScheduling 注解:

@SpringBootApplication
@EnableScheduling
public class DemoApplication {

public static void main(String[] args) {
SpringApplication.run(DemoApplication.class, args);
}

}

只有打上这个注解, SpringBoot 在启动后才会扫描带 @Scheduled 注解的方法,依次去执行。

新建两个定时任务,忽略命名等规范性问题,关注本文主题:

import org.springframework.scheduling.annotation.Scheduled;
import org.springframework.stereotype.Component;
import java.util.Date;

@Component("aa")
public class aa {
@Scheduled(cron = "0 59 17 * * ?")
public void bb() {
try {
System.out.println("aa执行时间:" + new Date());
Thread.sleep(60000);
System.out.println("aa完成时间:" + new Date());
} catch (Exception e) {
e.printStackTrace();
}
}
}
import org.springframework.scheduling.annotation.Scheduled;
import org.springframework.stereotype.Component;
import java.util.Date;

@Component("bb")
public class bb {
@Scheduled(cron = "10 59 17 * * ?")
public void aa() {
try {
System.out.println("bb执行时间:" + new Date());
Thread.sleep(5000);
System.out.println("bb完成时间:" + new Date());
} catch (Exception e) {
e.printStackTrace();
}
}
}

注意,我这里第一个定时任务是在下午的5点39分00秒执行,要执行60秒。第二个定时任务,本意是在下午5点39分10秒执行。但是由于单线程,第一个定时任务还未执行完毕,导致第二个定时任务顺延,无法按时执行,执行结果为:

aa执行时间:Fri Nov 06 17:59:00 CST 2020
aa完成时间:Fri Nov 06 18:00:00 CST 2020
bb执行时间:Fri Nov 06 18:00:00 CST 2020
bb完成时间:Fri Nov 06 18:00:05 CST 2020

如何解决呢?自然就是分配多个线程,让其不互相阻塞。

问题解决

较新的版本,是支持直接配置项来设定线程池大小的,很简单:

spring.task.scheduling.pool.size=10

或者用配置类的形式:

import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.scheduling.annotation.SchedulingConfigurer;
import org.springframework.scheduling.config.ScheduledTaskRegistrar;

import java.util.concurrent.Executor;
import java.util.concurrent.Executors;

@Configuration
public class ScheduledConfig implements SchedulingConfigurer {
@Override
public void configureTasks(ScheduledTaskRegistrar scheduledTaskRegistrar) {
scheduledTaskRegistrar.setScheduler(setTaskExecutors());
}

@Bean(destroyMethod="shutdown")
public Executor setTaskExecutors(){
return Executors.newScheduledThreadPool(10); // 10个线程来处理。
}
}

调整了下触发时间,运行结果为:

aa执行时间:Fri Nov 06 17:46:00 CST 2020
bb执行时间:Fri Nov 06 17:46:10 CST 2020
bb完成时间:Fri Nov 06 17:46:15 CST 2020
aa完成时间:Fri Nov 06 17:47:00 CST 2020

cron表达式

@Scheduled 有三种定时任务的执行方式,包括 fixedDelay 、 fixedRate 、 corn 表达式,下面就分别讲讲这三种执行方式的不同。

cron 一般是六个或七个字段,分别是:

1. Seconds (秒) 
2. Minutes (分)
3. Hours (时)
4. Day (每月的第几天,day-of-month)
5. Month (月)
6. Day (每周的第几天,day-of-week)
7. Year (年 可选字段)

每隔字段的范围以及特殊字符:

秒 :范围:0-59 
分 :范围:0-59
时 :范围:0-23
天(月) :范围:1-31,但要注意一些特别的月份2月份没有只能1-28,有些月份没有31
月 :用0-11 或用字符串 “JAN, FEB, MAR, APR, MAY, JUN, JUL, AUG, SEP, OCT, NOV and DEC” 表示
天(周):用1-7表示(1 = 星期日)或用字符口串“SUN, MON, TUE, WED, THU, FRI and SAT”表示
年:范围:1970-2099

“/”:表示为“每”,如“0/10”表示每隔10分钟执行一次,“0”表示为从“0”分开始, “3/20”表示表示每隔20分钟执行一次
“?”:只用于月与周,表示不指定值
“L”:只用于月与周,5L用在月表示为每月的最后第五天天;1L用在周表示每周的最后一天;
“W”::表示有效工作日(周一到周五),只能出现在day-of-month,系统将在离指定日期的最近的有效工作日触发事件。例如:在 DayofMonth使用5W,如果5日是星期六,则将在最近的工作日:星期五,即4日触发。如果5日是星期天,则在6日(周一)触发;如果5日在星期一到星期五中的一天,则就在5日触发。另外一点,W的最近寻找不会跨过月份
“#”:用于确定每个月第几个星期几,只能出现在DayofMonth域。例如在4#2,表示某月的第二个星期三。
“*” 代表整个时间段。

注意:每个元素可以是一个值(如6),一个连续区间(9-12),一个间隔时间(8-18/4)(/表示每隔4小时),一个列表(1,3,5),通配符。由于"月份中的日期"和"星期中的日期"这两个元素互斥的,必须要对其中一个设置‘?’

表达式实例:

0 0 10,14,16 * * ? 每天上午10点,下午2点,4点
0 0/30 9-17 * * ? 朝九晚五工作时间内每半小时
0 0 12 ? * WED 表示每个星期三中午12点
"0 0 12 * * ?" 每天中午12点
"0 15 10 ? * *" 每天上午10:15
"0 15 10 * * ?" 每天上午10:15
"0 15 10 * * ? *" 每天上午10:15
"0 15 10 * * ? 2005" 2005年的每天上午10:15
"0 * 14 * * ?" 在每天下午2点到下午2:59期间的每1分钟
"0 0/5 14 * * ?" 在每天下午2点到下午2:55期间的每5分钟
"0 0/5 14,18 * * ?" 在每天下午2点到2:55期间和下午6点到6:55期间的每5分钟
"0 0-5 14 * * ?" 在每天下午2点到下午2:05期间的每1分钟
"0 10,44 14 ? 3 WED" 每年三月的星期三的下午2:10和2:44
"0 15 10 ? * MON-FRI" 周一至周五的上午10:15
"0 15 10 15 * ?" 每月15日上午10:15
"0 15 10 L * ?" 每月最后一日的上午10:15
"0 15 10 ? * 6L" 每月的最后一个星期五上午10:15
"0 15 10 ? * 6L 2002-2005" 2002年至2005年的每月的最后一个星期五上午10:15
"0 15 10 ? * 6#3" 每月的第三个星期五上午10:15
Logo

魔乐社区(Modelers.cn) 是一个中立、公益的人工智能社区,提供人工智能工具、模型、数据的托管、展示与应用协同服务,为人工智能开发及爱好者搭建开放的学习交流平台。社区通过理事会方式运作,由全产业链共同建设、共同运营、共同享有,推动国产AI生态繁荣发展。

更多推荐