所使用的jar包
<dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-quartz</artifactId></dependency>
使用默认单机模式。单机模式中,Job 和Trigger是存放在内存中Map,通过源码可以看出 quartz-2.3.0.sources.jar!/org/quartz/simpl/RAMJobStore.java
protected HashMap<JobKey, JobWrapper> jobsByKey = new HashMap<JobKey, JobWrapper>(1000);protected HashMap<TriggerKey, TriggerWrapper> triggersByKey = new HashMap<TriggerKey, TriggerWrapper>(1000);protected HashMap<String, HashMap<JobKey, JobWrapper>> jobsByGroup = new HashMap<String, HashMap<JobKey, JobWrapper>>(25);protected HashMap<String, HashMap<TriggerKey, TriggerWrapper>> triggersByGroup = new HashMap<String, HashMap<TriggerKey, TriggerWrapper>>(25);protected TreeSet<TriggerWrapper> timeTriggers = new TreeSet<TriggerWrapper>(new TriggerWrapperComparator());protected HashMap<String, Calendar> calendarsByName = new HashMap<String, Calendar>(25);protected Map<JobKey, List<TriggerWrapper>> triggersByJob = new HashMap<JobKey, List<TriggerWrapper>>(1000);protected final Object lock = new Object();protected HashSet<String> pausedTriggerGroups = new HashSet<String>();protected HashSet<String> pausedJobGroups = new HashSet<String>();protected HashSet<JobKey> blockedJobs = new HashSet<JobKey>();
官网支持集群是把这些数据存放在mysql, 也有人改成使用redis存放这些数据
结合Springboot
注意此处Job要扩展QuartzJobBean, 只有这样才能使用@Autowired进来的其它service实例,否则要显式地new 一个相应service的实例
@Slf4j
@Component
@DisallowConcurrentExecution
public class MyJob extends QuartzJobBean {@AutowiredApplicationService applicationService;@Overrideprotected void executeInternal(JobExecutionContext jobExecutionContext) throws JobExecutionException {log.info("触发MyJob=========");applicationService.doSomething();Trigger trigger = jobExecutionContext.getTrigger();log.info("触发MyJob==========the trigger time : {}, 当前时间: {}",trigger.getStartTime(), new Date());}
}
设置Schedule
public interface QuartzService {void startSchedule ();void deployMySchedule (Myparams params) throws SchedulerException, ParseException;}
定义trigger并设置Schedule
@Slf4j
@Service
public class QuartzServiceImpl implements QuartzService {private final SimpleDateFormat formatter = new SimpleDateFormat("yyyy-MM-dd hh:mm:ss", Locale.ENGLISH);private final String myJobName = "MyJobName";private final String QUARTZ_JOB_GROUP_SUFFIX = "Group";@Autowiredprivate Scheduler scheduler;@Overridepublic void deployMySchedule (MyParams params) throws SchedulerException, ParseException {if (checkMyJobExists(params)) {log.warn("已存在{}的任务", params.getJobName());updateMyJobTrigger(params);} else {log.info("不存在{}的任务, 添加任务", params.getJobName());arrangeNewMyJobSchedule(params);}}@Overridepublic void startSchedule () {try {scheduler.start();} catch (SchedulerException e) {log.error("Quartz 启动Schedule 出现异常 ==== ", e);}}/*** {* "myJobName": "my.customized.job.name",* "triggerTime": "2023-09-21 23:23:23"* }* @param params* @throws SchedulerException*/private void arrangeNewMyJobSchedule (MyParams params) throws SchedulerException, ParseException {String jobName = params.getMyJobName();String jobGroup = params.getMyJobName()+QUARTZ_JOB_GROUP_SUFFIX;String triggerTime = params.getTriggerTime();// TODO change triggerTime to Date()Date date = formatter.parse(triggerTime);log.info("{} 添加定时任务", params.getMyJobName());JobKey theJobKey = jobKey(jobName, jobGroup);JobDetail job = JobBuilder.newJob(MyJob.class).usingJobData("myJobName", params.getMyJobName()).withIdentity(theJobKey).build();// Simple trigger without repeating// withMisfireHandlingInstructionNextWithRemainingCount()// Does nothing, misfired execution is ignored and there is no next execution.// Use this instruction when you want to completely discard the misfired execution.// Example scenario: the trigger was suppose to start recording of a program in TV.// There is no point of starting recording when the trigger misfired and is already 2 hours late.// Discarded but job not removedTrigger trigger = TriggerBuilder.newTrigger().withIdentity(triggerKey(jobName, jobGroup)).startAt(date).withSchedule(SimpleScheduleBuilder.simpleSchedule().withMisfireHandlingInstructionNextWithRemainingCount()).build();// 使用触发器调度任务的执行scheduler.scheduleJob(job, trigger);log.info("{} 设置任务触发时间为: {}, 配置触发器", params.getMyJobName(), triggerTime);log.info("schedule: scheduleName {}, scheduleInstanceId {} ", scheduler.getSchedulerName(),scheduler.getSchedulerInstanceId());}private void updateMyJobTrigger (MyParams params) throws SchedulerException, ParseException {String jobName = params.getMyJobName();String jobGroup = params.getMyJobName()+"Group";String triggerTime = params.getTriggerTime();updateJobTrigger(jobName, jobGroup, triggerTime);}private boolean checkMyJobExists (MyParams params) throws SchedulerException {String jobName = params.getMyJobName();String jobGroup = params.getMyJobName()+QUARTZ_JOB_GROUP_SUFFIX;JobKey jobKey = new JobKey(jobName, jobGroup);return scheduler.checkExists(jobKey);}private void updateJobTrigger (String jobName, String jobGroup, String triggerTime) throws ParseException {// TODO change triggerTime to Date()Date date = formatter.parse(triggerTime);try {TriggerKey triggerKey = triggerKey(jobName, jobGroup);SimpleTrigger oldTrigger = (SimpleTrigger) scheduler.getTrigger(triggerKey);// Simple trigger without repeating// withMisfireHandlingInstructionNextWithRemainingCount()// 如果给的时间小于当前时间, 只重新配置触发器, 并不触发, 同时 jobdetail 也没有删除Trigger newTrigger = oldTrigger.getTriggerBuilder().withIdentity(triggerKey).withSchedule(simpleSchedule().withMisfireHandlingInstructionNextWithRemainingCount()).startAt(date).build();// 重启触发器scheduler.rescheduleJob(oldTrigger.getKey(), newTrigger);log.info("Job {} 任务触发时间更新为: {}, 重新配置触发器", jobName, triggerTime);} catch (SchedulerException e) {e.printStackTrace();}}
}
在项目启动时加载schedule
@Slf4j
@Component
public class MyQuartzScheduleStart {@AutowiredQuartzService quartzService;@PostConstructpublic void init() {log.info("Quartz 调度任务开始 =====");quartzService.startSchedule();}
}