Java定时任务调度如何避免调度混乱

在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

你可能感兴趣的:(Java定时任务调度如何避免调度混乱)