在Java定时任务调度中,使用的最多的两种定时任务调度框架是Spring Scheduled 和 Quartz.目前JDK的Timer已经使用的很少了.
实际生产中,我们可能会遇到这样的问题:
定义某个定时任务A,然后每过5分钟运行一次,去扫描数据库做一个数据同步,但是这项任务在生产中实际执行的时间需要10分钟.
这样调度器在触发时间点调起一次A任务后,无论这次任务是否已经完成,5分钟之后,还会继续调起A任务,但是此时上一次任务还在运行同步中,这样就有可能会造成数据同步混乱.
如何解决呢?
正确的思路应该是:如果上一次调起的任务还没有完成,那么下一次调度器就不应该在调起这个任务了.
在Spring Scheduled中,我们可以通过ReentrantLock的tryLock()方法来进行判断上次任务十分已经完成.如果上次任务还没完成,那么就不再重复执行调度程序
@Scheduled("${cron}")
public void method() {
if (myReentrantLock.tryLock()) {
try {
//执行方法
} finally {
myReentrantLock.unlock();
}
}
}
在Quartz中,可以通过注解@DisallowConcurrentExecution的方式,来防止同一个JobDetail并发调度,原理也是如果上一次调度起来的任务还没有完成,就不在重复调度相同的JobDetail.
@DisallowConcurrentExecution
public class DefaultQuartzJobBean extends QuartzJobBean