很多情况下,我们的系统中需要使用定时任务。比如定时刷新,定时计算等等。
在spring boot中,执行定时任务,其实是很简单的。使用以下两个注解即可。
@EnableScheduling 激活定时任务。添加到spring boot启动类中即可
@scheduled:定时任务执行计划
虽然已经很熟悉如何定义一个执行计划。但是这里还是要记录一下表达式。
@scheduled注解分两种定时表达式:
延迟:
fixedDelay:上一次执行完毕时间点之后多长时间再执行
fixedDelayString:fixedDelay 意思相同,只是使用字符串的形式。支持占位符。
fixedRate:上一次开始执行时间点之后多长时间再执行
fixedRateString:fixedRate 意思相同,只是使用字符串的形式。支持占位符。
initialDelay:第一次延迟多长时间后再执行
initialDelayString:initialDelay 意思相同,只是使用字符串的形式。支持占位符
fixedDelay和fixedRate的区别就是:fixedDelay是以上一个任务的结束时间为开始点进行延迟。而fixedRate是以上一个任务的开始时间为开始点进行延迟。
如:@Scheduled(initialDelay=1000, fixedRate=5000)。@Scheduled(fixedDelayString = "${time.fixedDelay}")
定时:cron表达式
格式为:[秒] [分] [小时] [日] [月] [周] [年]
每隔5秒执行一次:*/5 * * * * ?
每隔1分钟执行一次:0 */1 * * * ?
每天23点执行一次:0 0 23 * * ?
在26分、29分、33分执行一次:0 26,29,33 * * * ?
以上配置完成以后,spring boot就会按照给定的时候,定时执行任务了。
这里要注意点,spring boot在执行定时任务的时候,默认是以单线程来执行的。即,所有任务都会在一个线程中执行。如果两个任务都定义在同一时间执行,那么会导致其中一个任务的执行时间会推后,会在上一个任务完成之后,该任务才会执行。
如何解决单线程的问题呢,那就是并发执行。大家网上查资料也都知道怎么解决了。那就是使用如下两个注解:
@EnableAsync:添加到spring boot启动类中即可,和@EnableScheduling一样
@Async:添加到@scheduled修饰的方法上,或者接口中都可以。
正常情况下,使用这两个注解,就可以解决定时任务单线程执行的问题,变为多线程执行。但是我在使用过程当中,发现了一个坑。那就是以上两个注解,在spring boot 2.0版本以上才可以多线程执行。在1.0版本上,只写这两个注解,还是会以单线程来执行。解决方法如下:
@Bean
public ThreadPoolTaskScheduler threadPoolTaskScheduler(){
ThreadPoolTaskScheduler executor = new ThreadPoolTaskScheduler();
executor.setPoolSize(5);
executor.setWaitForTasksToCompleteOnShutdown(true);
executor.setAwaitTerminationSeconds(60);
executor.setRejectedExecutionHandler(new ThreadPoolExecutor.CallerRunsPolicy());
return executor;
}
定义执行任务的线程池。添加以上配置,即可在1.0版本下进行多线程的定时任务执行了。
@scheduled底层使用的是spring的任务调度线程池ThreadPoolTaskScheduler,而ThreadPoolTaskScheduler定义的线程池为1个。