我在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);
}
});
}
}
答案 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
}