最原始试用方法如下:
@Service
@Slf4j
public class ScheduleService {
private Random random = new Random();
@Scheduled(cron = "*/5 * * * * ?")
public void run() throws InterruptedException {
int taskId = random.nextInt(100);
log.debug("taskId for run 1: {} start at {}", taskId, new Date());
Thread.sleep(7000);
log.debug("taskId for run 1: {} end at {}", taskId, new Date());
}
}
其实cron的表达式范例:
每隔5秒执行一次:*/5 * * * * ?
每隔1分钟执行一次:0 */1 * * * ?
每天23点执行一次:0 0 23 * * ?
每天凌晨1点执行一次:0 0 1 * * ?
每月1号凌晨1点执行一次:0 0 1 1 * ?
每月最后一天23点执行一次:0 0 23 L * ?
每周星期天凌晨1点实行一次:0 0 1 ? * L
在26分、29分、33分执行一次:0 26,29,33 * * * ?
每天的0点、13点、18点、21点都执行一次:0 0 0,13,18,21 * * ?
不过经过测试发现,默认的Scheduled函数为同步执行(即上次执行未完成的情况下下一次不会再次运行)
要变成异步执行的方法很简单,在spring boot启动类上面加上EnableAsync, 并在需要异步的函数头上加上Async即可。
如果有多个scheduled注解的函数:
@Service
@Slf4j
public class ScheduleService {
private Random random = new Random();
@Scheduled(cron = "*/5 * * * * ?")
public void run1() throws InterruptedException {
//默认同步执行,如果要改成异步,需要在启动类加enableAsync,并且在需要异步的函数头加上Async.
int taskId = random.nextInt(100);
log.debug("taskId for run 1: {} start at {}", taskId, new Date());
Thread.sleep(7000);
log.debug("taskId for run 1: {} end at {}", taskId, new Date());
}
@Scheduled(cron = "*/10 * * * * ?")
public void run2() throws InterruptedException {
int taskId = random.nextInt(100);
log.debug("taskId for run 2: {} start at {}", taskId, new Date());
Thread.sleep(7000);
log.debug("taskId for run 2: {} end at {}", taskId, new Date());
}
}
则会发现,所有的函数都在同一个线程里面执行,不符合原本的业务需求,如果要改成函数相互异步自身同步的方法是配置线程池中的线程数:
@Configuration
public class ScheduleConfiguration implements SchedulingConfigurer {
@Override
public void configureTasks(ScheduledTaskRegistrar taskRegistrar) {
taskRegistrar.setScheduler(Executors.newScheduledThreadPool(5));
}
}
如上图配置即可保证5个以下的schedule函数相互之间不影响。
另外,如果要在test里面里面取消schedule函数的自动运行,需要单独抽一个配置类来配置@EnableScheduling
并设定ConditionalOnProperty,如下代码:
@Configuration
@EnableScheduling
@ConditionalOnProperty(name = "参数值", havingValue = "对比值")
public class ScheduleConfiguration implements SchedulingConfigurer {
}