Quartz暂停后再恢复时job执行多次

问题描述:Quartz暂停后,恢复出现多次执行Job,下面是源码操作

1、Quartz创建一个Job任务,这个很简单

/**
     * 新增/修改任务
     * @param quartz
     * @return
     */
    @Override
    public Result saveJob(QuartzJob quartz,Object obj){
        try {
            //如果是修改  展示旧的 任务
            if(quartz.getOldJobGroup() != null && !"".equals(quartz.getOldJobGroup())){
                JobKey key = new JobKey(quartz.getOldJobName(),quartz.getOldJobGroup());
                scheduler.deleteJob(key);
            }
            //构建job信息
            //构建job信息
            Class cls = null;
            //第一种方式:从页面传入值 如:包+类 包名获取class对象
            if (StringUtils.isNotEmpty(quartz.getJobClassName())){
                cls = Class.forName(quartz.getJobClassName());
            }
            //第二种方式:直接使用obj对象获取class
            if (cls==null&&obj!=null){
                cls = obj.getClass();
            }
            cls.newInstance();
            JobDetail job = JobBuilder.newJob(cls).withIdentity(quartz.getJobName(),
                    quartz.getJobGroup())
                    .withDescription(quartz.getDescription()).build();
            //添加参数
            job.getJobDataMap().put("jobParam", quartz.getJobDataParam());
            // 触发时间点
            CronScheduleBuilder cronScheduleBuilder = CronScheduleBuilder.cronSchedule(quartz.getCronExpression().trim());
            cronScheduleBuilder.withMisfireHandlingInstructionDoNothing();
            Trigger trigger = TriggerBuilder.newTrigger().withIdentity("trigger"+quartz.getJobName(), quartz.getJobGroup())
                    .startNow().withPriority(quartz.getPriority()).withSchedule(cronScheduleBuilder).build();
            //交由Scheduler安排触发
            scheduler.scheduleJob(job, trigger);
        } catch (Exception e) {
            e.printStackTrace();
            return Result.error();
        }
        return Result.ok();
    }

2、暂停Job任务

/**
     * 停止任务
     * @param jobName
     * @param jobGroup
     * @return
     */
    @Override
    public Result pauseJob(String jobName, String jobGroup) {
        JobKey key = new JobKey(jobName,jobGroup);
        try {
            scheduler.pauseJob(key);
        } catch (SchedulerException e) {
            e.printStackTrace();
            return Result.error();
        }
        return Result.ok();
    }```
3、恢复Job任务

```java
/**
     * 恢复任务
     * @param jobName
     * @param jobGroup
     * @return
     */
    @Override
    public Result resumeJob(String jobName, String jobGroup) {
        JobKey key = new JobKey(jobName,jobGroup);
        try {
            scheduler.resumeJob(key);
        } catch (SchedulerException e) {
            e.printStackTrace();
            return Result.error();
        }
        return Result.ok();
    }

结果:
在这里插入图片描述
暂停是没问题,恢复Job就出现多次执行Job情况,经过定位和查找资料,发现Quartz暂停后 恢复启动 为什么会执行多次 会把暂停中间需要执行的次数在恢复启动时一次补偿执行完毕
cronScheduleBuilder.withMisfireHandlingInstructionDoNothing();因为这正是WithMisfireHandlingInstructionIgnoreMisfires()所要求的,重做所有错过的周期。
但是知道原理,还是要解决现实问题。
quartz.properties中还需要配置这个参数值,要正确的值,如下:

#这个时间大于1000010秒)会导致MISFIRE_INSTRUCTION_DO_NOTHING不起作用。  
org.quartz.jobStore.misfireThreshold = 6000

好啦,问题保管解决,不解决你找我!

解释下这个配置设置的意义:

misfireThreshold表示实际执行时间与下一次应该执行时间之间的差值,超过这个差值就不会执行,低于这个差值就会执行。

比如我每3秒执行一次,misfireThreshold=6000,当你暂停低于6秒内,它会弥补执行,超过6秒,它就不再弥补执行了。

Quartz 中,一个 Job 可能会被多个 Trigger 触发执行。但是,如果你想要一个 Trigger 触发执行多个 Job,可以使用 QuartzJobChainingJobListener 来实现。 JobChainingJobListener 允许我们在一个 Job 执行完成后自动触发另一个 Job执行。这样,你就可以在一个 Trigger 触发的情况下执行多个 Job。 以下是一个实现示例: ```java // 创建 JobDetail 对象 JobDetail job1 = JobBuilder.newJob(Job1.class).withIdentity("job1").build(); JobDetail job2 = JobBuilder.newJob(Job2.class).withIdentity("job2").build(); // 创建 Trigger 对象 Trigger trigger1 = TriggerBuilder.newTrigger().withIdentity("trigger1") .startNow().build(); // 创建 JobChainingJobListener 对象 JobChainingJobListener listener = new JobChainingJobListener("listener"); listener.addJobChainLink(job1.getKey(), job2.getKey()); // 将 listener 注册到 Scheduler 上 scheduler.getListenerManager().addJobListener(listener); // 将 JobDetail 和 Trigger 注册到 Scheduler 上 scheduler.scheduleJob(job1, trigger1); ``` 在上面的示例中,我们创建了两个 JobJob1 和 Job2。然后,我们创建了一个 Trigger,并将其与 Job1 关联。接下来,我们创建了一个 JobChainingJobListener 对象,并将其注册到 Scheduler 上。最后,我们将 Job1 和 Trigger 注册到 Scheduler 上。 当 Trigger 触发 Job1 的执行JobChainingJobListener 会自动触发 Job2 的执行。这样,一个 Trigger 就可以触发多个 Job执行了。
评论 5
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值