这是我参与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(){
.......
}
}