Spring定时任务@Scheduled用法

1,115 阅读4分钟

这是我参与8月更文挑战的第4天,活动详情查看:8月更文挑战

使用目的

@Scheduled由Spring定义,用于将方法设置为调度任务,如:固定时间点执行某个任务,每隔多少时间被执行等等。

基本用法

1、@Scheduled(fixedDelay=1000)

上一个任务结束到下一个任务开始的时间间隔为固定的1秒,任务的执行总是要先等到上一个任务的执行结束。

2、@Scheduled(fixedRate=1000)

每间隔1秒钟就会执行任务(如果任务执行的时间超过一秒,则下一个任务在上一个任务结束之后立即执行)

3、@Scheduled(fixedDela=1000,initialelay=2000)

第一次执行的任务将会延迟2秒钟后才会启动

4、@Scheduled(cron="0 15 8 * * ?")

Corn表达式,每个月的15号上午8点开始执行任务,生成方式可以访问cron.qqe2.com/生成。

5、在配置文件中配置任务调度的参数

@Scheduled(fixedDelayString = "fixedDelay.in.milliseconds") 

@Scheduled(fixedRateString="fixedDelay.in.milliseconds")

@Scheduled(fixedRateString="{fixedRate.in.milliseconds}") 

@Scheduled(cron = "${cron.expression}")

Corn表达式

Corn表达式是一个字符串,字符串以5或6个空格隔开,分为6或7个域,每一个域代表一个含义,Corn有如下两种语法格式:

(1)Second Minutes Hours DayofMonth Month DayofWeek Year

(2)Second Minutes Hours DayofMonth Month DayofWeek

字段允许值允许的特殊字符
秒(Seconds)0-59的整数,-*/四个字符
分(Minutes)0-59的整数,-*/四个字符
小时(Hours)0-23的整数,-*/四个字符
日期(DayofMonth)1-31的整数(要考虑当月的天数),- * ? / L W C     八个字符
月份(Month)1-12的整数JAN-DEC,-*/四个字符
星期(DayofWeek)1-7的整数或者SUN-SAT(1-SUN),- * ? / L W C     八个字符
年(可选,留空)(Year)1970-2099,-*/四个字符
  • :表示匹配该域的任意值,假如Minutes域使用,即表示每分钟都会触发事件。
  • ?:只能用在DayofMonth和DayofWeek两个域,它也匹配域的任意值,但实际不会,因为DayofMonth和DayofWeek会相互影响。例如想在每月的20日触发调度,不管20日到底是星期几,则只能使用如下写法:13 13 15 20 * ? ,其中最后一位只能用?,而不能使用*,如果使用*表示不管星期几都会触发,实际上并不是这样。
  • -:表示范围,例如Minutes域使用5-20,表示5分到20分钟每分钟触发一次
  • /:表示起始时间开始触发,然后每隔固定时间触发一次,例如在minutes域使用5/20,则意味着5分钟触发一次,而25,45等分别触发一次。
  • ,:表示列出枚举值,例如:在minutes域使用5,20,则意味着在5和20分每分钟触发一次。
  • L:表示最后,只能出现在DayofWeek和DayofMonth域,如果DayofWeek域使用5L,意味着在最后的一个星期四触发
  • W:表示有效工作日(周一到周五),只能出现在DayofMonth域,系统将在离指定日期的最近的有效工作日触发事件。例如在DayofMonth使用5W,如果5日是星期六。则将在最近的工作日:星期五,即4日触发。如果5日是星期天,则在6日(周一)触发,如果5日在星期一到星期五中的一天,则就在5日触发,另外一点,W的最近寻找不会跨过月份。
  • LW:这两个字符可以连用,表示在某个月最后一个工作日,即最后一个星期五。
  • #:用于确定每个月第几个星期几,只能出现在DayofMonth域,例如4#2,表示某月的第二个星期三

线程

spring scheduled默认是所有定时任务都在一个线程中执行,也就是说定时任务1一直在执行,定时任务2一直在等待定时任务1执行完成,为了避免相互等待的情况,可以为定时任务配置线程池。

@Configuration
public class ScheduleConfig implements SchedulingConfigurer { 
@Override 
public void configureTasks(ScheduledTaskRegistrar taskRegistrar) { 
    taskRegistrar.setScheduler(Executors.newScheduledThreadPool(50));
 } 
}

在程序启动后,会逐步启动50个线程,放在线程池中,每个定时任务会占用1个线程,但是相同的定时任务,执行的时候,还是会在同一个线程中。

也可以配置异步执行,相同的任务也不会影响

添加配置

@Configuration 
@EnableAsync 
public class ScheduleConfig {
 @Bean 
 public TaskScheduler taskScheduler() { 
         ThreadPoolTaskScheduler taskScheduler = new ThreadPoolTaskScheduler(); 
         taskScheduler.setPoolSize(50); scheduler.setThreadNamePrefix("scheduled-thread-"); 
         //设置线程池关闭的时候等待所有任务都完成再继续销毁其他的Bean 
         scheduler.setWaitForTasksToCompleteOnShutdown(true);
         //设置线程池中任务的等待时间,如果超过这个时候还没有销毁就强制销毁,以确保应用最后能够被关闭,而不是阻塞住 
         scheduler.setAwaitTerminationSeconds(60); 
         //这里采用了CallerRunsPolicy策略,当线程池没有处理能力的时候,该策略会直接在 execute 方法的调用线程中运行被拒绝的任务;如果执行程序已关闭,则会丢弃该任务 
         scheduler.setRejectedExecutionHandler(new ThreadPoolExecutor.CallerRunsPolicy()); return taskScheduler; 
     } 
   }

在方法上添加注解@Async

public class TaskFileScheduleService {
 @Async 
 @Scheduled(cron="0 */1 * * * ?")
 public void task1(){ 
     .......
     }
 @Async 
 @Scheduled(cron="0 */1 * * * ?") 
 public void task2(){ 
     ....... 
     } 
   }