springboot计划任务@EnableScheduling和@Scheduled
@Scheduled中的参数说明
- @Scheduled(fixedRate=2000):上一次开始执行时间点后2秒再次执行;
- @Scheduled(fixedDelay=2000):上一次执行完毕时间点后2秒再次执行;
- @Scheduled(initialDelay=1000, fixedDelay=2000):第一次延迟1秒执行,然后在上一次执行完毕时间点后2秒再次执行;
- @Scheduled(cron=“* * * * * ?”):按cron规则执行。
- service
package ch2.scheduler; //时间处理,时间格式化 import java.util.Date; import java.text.SimpleDateFormat; //spring计划任务声明(针对方法声明) import org.springframework.scheduling.annotation.Scheduled; //spring组件声明 import org.springframework.stereotype.Service; //组件什么 @Service public class SchedulerService { //创建日期模板 private static final SimpleDateFormat dateFormat = new mpleDateFormat("HH::MM::ss"); //每5秒钟执行一次 @Scheduled(fixedRate = 5000) public void reportCurrentTime() { System.out.println("每五秒执行一次: " + dateFormat.format( new Date() )); } //按照cron属性指定执行时间:秒分时 @Scheduled(cron = "0 34 18 ? * *") public void fixTimeExecution() { System.out.println("在指定的时间内执行: " + dateFormat.format( new Date()) ); } }
- config
package ch2.scheduler; //自动引入包 import org.springframework.context.annotation.ComponentScan; //配置类声明 import org.springframework.context.annotation.Configuration; //计划任务类声明 import org.springframework.scheduling.annotation.EnableScheduling; //配置类声明 @Configuration //自动引入包 @ComponentScan("ch2.scheduler") //通过@EnableScheduling开启对计划任务的支持 @EnableScheduling public class TaskSchedulerConfig { }
- main
package ch2.scheduler; //引入容器 import org.springframework.context.annotation.AnnotationConfigApplicationContext; public class Main { public static void main(String[] args) { AnnotationConfigApplicationContext context = new AnnotationConfigApplicationContext(TaskSchedulerConfig.class); //SchedulerService schedulerService = context.getBean(SchedulerService.class); } }
- 注意:SpringBoot使用@scheduled定时执行任务的时候是在一个单线程中,如果有多个任务,其中一个任务执行时间过长,则有可能会导致其他后续任务被阻塞直到该任务执行完成。也就是会造成一些任务无法定时执行的错觉
- 测试
@Scheduled(cron = "0/1 * * * * ? ") public void deleteFile() throws InterruptedException { log.info("111delete success, time:" + new Date().toString()); Thread.sleep(1000 * 5);//模拟长时间执行,比如IO操作,http请求 } @Scheduled(cron = "0/1 * * * * ? ") public void syncFile() { log.info("222sync success, time:" + new Date().toString()); }
syncFile被阻塞了,直达deleteFile执行完它才执行了
而且从日志信息中也可以看出@Scheduled是使用了一个线程池中的一个单线程来执行所有任务的。
把Thread.sleep(1000*5)注释了,再次测试
- 解决
- 将@Scheduled注释的方法内部改成异步执行
//构建一个合理的线程池也是一个关键,否则提交的任务也会在自己构建的线程池中阻塞 ExecutorService service = Executors.newFixedThreadPool(5); @Scheduled(cron = "0/1 * * * * ? ") public void deleteFile() { service.execute(() -> { log.info("111delete success, time:" + new Date().toString()); try { Thread.sleep(1000 * 5);//改成异步执行后,就算你再耗时也不会印象到后续任务的定时调度了 } catch (InterruptedException e) { e.printStackTrace(); } }); } @Scheduled(cron = "0/1 * * * * ? ") public void syncFile() { service.execute(()->{ log.info("222sync success, time:" + new Date().toString()); }); }
- 把Scheduled配置成成多线程执行
@Configuration public class ScheduleConfig implements SchedulingConfigurer { @Override public void configureTasks(ScheduledTaskRegistrar taskRegistrar) { //当然了,这里设置的线程池是corePoolSize也是很关键了,自己根据业务需求设定 taskRegistrar.setScheduler(Executors.newScheduledThreadPool(5)); /**为什么这么说呢? 假设你有4个任务需要每隔1秒执行,而其中三个都是比较耗时的操作可能需要10多秒,而你上面的语句是这样写的: taskRegistrar.setScheduler(Executors.newScheduledThreadPool(3)); 那么仍然可能导致最后一个任务被阻塞不能定时执行 **/ } }
常用Cron表达式( 秒/分/时/日/月/周/年 )
- 0 0 10,14,16 * * ? 每天上午10点,下午2点,4点
- 0 0/30 9-17 * * ? 朝九晚五工作时间内每半小时
- 0 0 12 ? * WED 表示每个星期三中午12点
- “0 0 12 * * ?” 每天中午12点触发
- “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触发
- “0 15 10 ? * *” 每天上午10:15触发
- “0 15 10 * * ?” 每天上午10:15触发
- “0 15 10 * * ? *” 每天上午10:15触发
- “0 15 10 * * ? 2005” 2005年的每天上午10:15触发