Spring更新调度程序

时间:2016-12-16 14:34:14

标签: spring spring-boot scheduler spring-scheduled

我在Spring有预定的工作,我从我的数据库中获取了它的cron。 每次执行时,都会更新下一个执行时间。因此,如果将其配置为每10分钟运行一次,我可以将值更改为数据库,以便每15分钟安排一次该作业。

问题是我必须等待执行以获取更新的cron:如果每15分钟安排一个作业,并且我想将此值更改为每2分钟一次,我必须等待下一次执行(每隔2分钟就能完成这项工作。

有没有办法在我更新数据库后重新安排这份工作?

我想要销毁并刷新这个bean,但是它没有用(可能它不可能或我的实现中出错了)。也许有一种方法可以触发事件来执行方法configureTask。

这是我预定工作的片段。

@EnableScheduling
@Component
public class MyClass implements SchedulingConfigurer {

    private static final String JOB = "My personal task";

    @Autowired
    JobRepository jobRepository;

    @Override
    public void configureTasks(ScheduledTaskRegistrar scheduledTaskRegistrar) {
        scheduledTaskRegistrar.addTriggerTask(new Runnable() {
            @Override
            public void run() {
                System.out.println("Hello World!");
            }
        }, new Trigger() {
            @Override
            public Date nextExecutionTime(TriggerContext triggerContext) {
                JobScheduled byJobNameIgnoreCase = jobRepository.findByJobNameIgnoreCase(JOB); // read from database 
                String cron = byJobNameIgnoreCase.getCrontab();
                CronTrigger trigger = new CronTrigger(cron);
                return trigger.nextExecutionTime(triggerContext);
            }
        });
    }

}

2 个答案:

答案 0 :(得分:1)

为了解决这个问题,我创建了一个SchedulerOrchestrator来管理我的工作。这些作业包含SchedulerFuture。

我希望这些代码可以帮助其他人。

让我们从一个将由我的工作实现的界面开始:

public interface SchedulerObjectInterface {    
    void start();
    void stop();
}

每个作业都需要ScheduledFuture来停止并需要自动装配要安排的TaskScheduler。这里是一个作业的样本(您可以根据需要创建多个):

@Component
public class MyFirstJob implements SchedulerObjectInterface {

    private static final Logger log = LoggerFactory.getLogger(MyFirstJob.class);

    public static final String JOB = "MyFirstJob";

    @Autowired
    JobRepository jobRepository;

    private ScheduledFuture future;

    @Autowired
    private TaskScheduler scheduler;


    @Override
    public void start() {
        future = scheduler.schedule(new Runnable() {
            @Override
            public void run() {
                System.out.println(JOB + "  Hello World! " + new Date());
            }
        }, new Trigger() {
            @Override
            public Date nextExecutionTime(TriggerContext triggerContext) {
                String cron = cronConfig();
                System.out.println(cron);
                CronTrigger trigger = new CronTrigger(cron);
                return trigger.nextExecutionTime(triggerContext);
            }
        });

    }

    @Override
    public void stop() {
        future.cancel(false);
    }

    // retrieve cron from database
    private String cronConfig() {
        JobScheduled byJobNameIgnoreCase = jobRepository.findByJobNameIgnoreCase(JOB);
        return byJobNameIgnoreCase.getCrontab();
    }

}

最后,我们可以将我们的工作添加到协调器:

@Configuration
public class SchedulerOrchestrator {

    private static final Logger log = LoggerFactory.getLogger(SchedulerOrchestrator.class);

    private static Map<String, SchedulerObjectInterface> schduledJobsMap = new HashMap<>();

    @Autowired
    JobRepository jobRepository;

    @Autowired
    MyFirstJob myFirstJob;

    @Autowired
    MySecondJob mySecondJob;

    @Autowired
    TaskScheduler scheduler;

    @PostConstruct
    public void initScheduler() {
        schduledJobsMap.put(MyFirstJob.JOB, myFirstJob);
        schduledJobsMap.put(MySecondJob.JOB, mySecondJob);

        startAll();
    }

    public void restart(String job) {
        stop(job);
        start(job);
    }

    public void stop(String job) {
        schduledJobsMap.get(job).stop();
    }

    public void start(String job) {
        schduledJobsMap.get(job).start();
    }

    public void startAll() {
        for (SchedulerObjectInterface schedulerObjectInterface : schduledJobsMap.values()) {
            schedulerObjectInterface.start();
        }
    }

    @Bean
    public TaskScheduler scheduler() {
        return new ThreadPoolTaskScheduler();
    }
}

答案 1 :(得分:0)

考虑这种方法。您可以检查每一分钟(或另一个精确度)的实际时刻,而不是添加和删除计划任务,并立即执行必要的任务。这会更容易。检查Quartz Scheduler,它的CronExpression有isSatisfiedBy(Date date)方法。

@Scheduled(cron = "5 * * * * *) // do not set seconds to zero, cause it may fit xx:yy:59
public void runTasks() {
     LocalTime now = LocalTime.now(); // or Date now = new Date();
    // check and run
}