1.@Scheduled注解
在SpringBoot项目中使用定时任务时可以使用@Scheduled
标注在需要定时执行的方法上。该注解位于spring-context.jar
包中,关于@Scheduled的具体描述如下:
属性 | 说明 |
cron():String | 使用Cron表达式创建定时任务,值可以是字符串也 |
zone():String | 指定cron的时区,默认是空字符串,表示本地时区 |
fixedDelay():long | 任务执行的时间间隔,表示第一次任务执行完毕和第二次任务开始之间的时间,单位:毫秒 |
fixedRate():long | 每隔多久执行一次任务,第一次任务开始和第二次任务开始之间的时间,单位:毫秒 |
initialDelay():long | 表示第一次执行 |
注意:@Scheduled
注解要生效还需要在系统启动类或配置类上添加@EnableScheduling
注解
2.简单使用@Scheduled注解
2.1 首先这里创建了一个普通的SpringBoot项目叫SpringbootApplication,在启动类上添加@EnableSchedling注解
@SpringBootApplication
@EnableScheduling
public class SpringbootApplication {
public static void main(String[] args) {
SpringApplication.run(SpringbootApplication.class, args);
}
}
2.2 创建定时任务调度类TestSchedule
,并定义taskSchedule1
方法使用@Scheduled(cron = "0/10 * * * * ?")
标注,表示该方法从0
秒开始,每隔10
秒执行一次,方法内部获取了执行当前任务的线程,打印了任务开始和结束的时间
、线程ID
、线程的名字
,sleep
5秒
表示任务执行完成需要花费5秒
时间。使用@Component
注解,交给Spring容器管理。
@Component
public class TestSchedule {
@Scheduled(cron = "0/10 * * * * ?")
public void taskSchdule1() throws InterruptedException {
Thread t = Thread.currentThread();
System.out.println("taskSchule1 "+ DateTimeUtils.dateToTimeStr(new Date()) +" ThreadID:"+ t.getId() +" "+t.getName());
Thread.sleep(5000);
System.out.println("taskSchule1 end "+ DateTimeUtils.dateToTimeStr(new Date()) +" ThreadID:"+ t.getId() +" "+t.getName());
}
}
2.3 运行结果
到这里最简单的定时任务调度就算完成了,启动项目,打印结果如下:
运行结果正常,每隔10秒执行一次,每次执行花费5秒时间。
3. 第一个坑
我们现在在TestSchedule类中添加第二个需要任务调度方法,每隔3秒执行一次。
@Scheduled(cron = "0/3 * * * * ?")
public void taskSchdule2(){
Thread t = Thread.currentThread();
System.out.println("taskSchule2 "+ DateTimeUtils.dateToTimeStr(new Date()) +" ThreadID:"+ t.getId() +" "+t.getName());
}
再次运行,结果如下:
这里我们发现任务2,在a、b两处时间相差7秒已超过3秒,显然b处是在任务1结束之后立刻执行的,并且任务1和任务2都是同一个线程执行的。
因为: Spring中@EnableScheduling和@Scheduled标注的定时任务默认是单线程执行的
,这里任务1执行任务需要花费较长时间,所有阻塞了任务2的执行。
4. 使用@Async和@EnableAsync异步执行任务
事实上在Spring的定时任务包中提供了@EnableAsync
和@Async
注解用于多线程异步执行任务。
首先在启动类上添加@EnableAsync
注解,并在TestSchedule
类上标注@Async
注解,表示该类中所有标注了@Scheduled
的方法都使用异步处理方式。
再次运行项目,结果如下:
此时,任务1和任务2均运行正常,并且任务1和任务2都是不同线程在执行,不会出现任务之间相互阻塞的情况。
这里是解决了第一个坑的问题,但是实际上可能引入第二个坑。
5. 第二个坑
这里我们稍作修改将任务的睡眠时间改成11秒
Thread.sleep(11000);
,此时任务1的执行时间已经超过了它的调度时间。再次运行程序结果如下:
观察发现任务2正常执行,但是任务1中a、b和c、d两组出现了交叉,两组是不同线程执行的,因为任务1的执行时间超过了调度时间,所以,a处开始执行,在未执行完成的情况下,任务的调度时间到了,其他线程有立马调度了任务从c处开始执行。
这是使用@EnableAsync
和@Async
可能会出现的问题。
6. 解决坑1和坑2
再次修改代码,去掉5中的@EnableAsync
和@Async
注解,去掉2中的@EnableScheduling
注解,
创建一个任务配置类ScheduleConfig
实现SchedulingConfigurer
接口的configureTasks
方法,使用参数taskRegistrar为任务调度创建线程池
@Configuration
@EnableScheduling
public class ScheduleConfig implements SchedulingConfigurer {
@Override
public void configureTasks(ScheduledTaskRegistrar taskRegistrar) {
taskRegistrar.setScheduler(taskExecutor());
}
@Bean(destroyMethod="shutdown")
public Executor taskExecutor() {
return Executors.newScheduledThreadPool(10);
}
}
运行程序结果如下:
现在任务1和任务2均运行正常,并且任务1不会出现坑2中的交叉现象,任务1第二次调度会等到第一次调度执行完毕后地下一个调度时间点才会执行。
总结
SpringBoot中可以使用@EnableScheduling
和@Scheduled
注解实现定时任务调度,但是注意默认所有任务都被单个线程调度的,有可能任务之间发生阻塞现象,可以使用@EnableAsync
和@Async
注解实现异步多线程任务调度,但需要注意任务执行时间如果大于任务调度周期时间,可能出现同一个任务交叉执行的情况。当然也可以使用第6步中的方法避免上述问题。?