springboot集成quartz实现动态任务调度

quartz是一个开源的作业调度框架,本文就是介绍下springboot框架下继承quartz的一些使用示例

首先我们需要添加quartz的spring-boot-starter-quartz依赖

        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-quartz</artifactId>
        </dependency>

我们需要做一些配置。quartz提供了基于内存(MEMORY)和基于jdbc的两种任务存储方式。具体的配置可参考下面的示例

spring:
  quartz:
    jdbc:
      initialize-schema: never
    job-store-type: MEMORY
    properties:
      org:
        quartz:
          scheduler:
            instanceName: myQuartzCluster
            instanceId: AUTO
          jobstore:
            class: org.quartz.impl.jdbcjobstore.JobStoreTX
            driverDelegateClass: org.quartz.impl.jdbcjobstore.StdJDBCDelegate
            tablePrefix: qrtz_
            isClustered: true
            clusterCheckinInterval: 10000
            useProperties: false
          threadPool:
            class: org.quartz.simpl.SimpleThreadPool
            threadCount: 5
            threadPriority: 5
            threadsInheritContextClassLoaderOfInitializingThread: true

需要注意的是,如果是集群部署而且定时任务同时只能一个实例运行的话,需要配置成jdbc方式,这样就可以用到数据库的锁。MEMORY这种方式建议用于开发环境测试,不建议用于生产环境。

如果是使用jdbc的方式,quartz提供了不同数据库的建表语句。可以在quartz的jar包中找到,具体路径是org.quartz.impl.jdbcjobstore

quartz_ddl.png

定时任务的类需要继承自QuartzJobBean并重写 executeInternal方法。

package com.xchaset.quartz.schedule.task;

import com.xchaset.quartz.annotation.MyScheduled;
import lombok.extern.slf4j.Slf4j;
import org.quartz.DisallowConcurrentExecution;
import org.quartz.JobExecutionContext;
import org.quartz.JobExecutionException;
import org.springframework.scheduling.quartz.QuartzJobBean;
import org.springframework.stereotype.Service;

@Service
@Slf4j
@DisallowConcurrentExecution
@MyScheduled(name = "EmailTask",description = "email schedule task",cronExpression = "0 0/2 * * * ?")
public class EmailTask extends QuartzJobBean {

    @Override
    protected void executeInternal(JobExecutionContext jobExecutionContext) throws JobExecutionException {
        log.info("{} is execute!",this.getClass().getSimpleName());
    }
}

@DisallowConcurrentExecution注解可以控制我们的相同类的定时任务同时只有一个执行。通常的场景是如果定时任务实际执行时间大于设定的定时任务执行间隔时间,就会导致同时多个任务并行。

@MyScheduled是自定义的一个注解,主要用于应用启动时将定时任务注册。

package com.xchaset.quartz.annotation;

import org.springframework.stereotype.Service;

import java.lang.annotation.*;

@Documented
@Target(ElementType.TYPE)
@Retention(RetentionPolicy.RUNTIME)
@Service
public @interface MyScheduled {

    String name() default "";

    String description() default "";

    String cronExpression() default "";

    boolean enable() default true;

}

获取注解类上的信息

package com.xchaset.quartz.utils;

import com.xchaset.quartz.annotation.MyScheduled;
import com.xchaset.quartz.schedule.JobBean;
import org.springframework.core.annotation.AnnotationUtils;
import org.springframework.scheduling.quartz.QuartzJobBean;

import java.util.ArrayList;
import java.util.List;
import java.util.Map;


public class MyAnnotationUtil {

    public static List<JobBean> getAllMyScheduledTask(){
        List<JobBean> jobBeans = new ArrayList<>();
        Map<String, QuartzJobBean> beansOfType = SpringContextHolder.getBeansOfType(QuartzJobBean.class);
        for (Map.Entry<String, QuartzJobBean> beanEntry : beansOfType.entrySet()) {
            MyScheduled annotation = AnnotationUtils.findAnnotation(beanEntry.getValue().getClass(), MyScheduled.class);
            QuartzJobBean value = beanEntry.getValue();
            if (annotation == null) {
                continue;
            }
            if (annotation.enable()) {
                String name = annotation.name();
                String description = annotation.description();
                String cronExpression = annotation.cronExpression();
                JobBean build = JobBean.builder().jobName(name)
                        .groupName("myGroup")
                        .cronExpression(cronExpression)
                        .description(description)
                        .quartzJobBean(value.getClass())
                        .build();
                jobBeans.add(build);
            }
        }
        return jobBeans;
    }
}

启动时注册任务

package com.xchaset.quartz.schedule;

import com.xchaset.quartz.utils.MyAnnotationUtil;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.CommandLineRunner;
import org.springframework.stereotype.Component;

import java.util.List;

@Component
public class ScheduleCommandLine implements CommandLineRunner {

    @Autowired
    private QuartzService quartzService;

    @Override
    public void run(String... args) throws Exception {
        List<JobBean> allMyScheduledTask = MyAnnotationUtil.getAllMyScheduledTask();
        for (JobBean jobBean : allMyScheduledTask) {
            quartzService.deleteJob(jobBean.getGroupName(),jobBean.getJobName());
            quartzService.addJob(jobBean.getQuartzJobBean(),jobBean.getGroupName(),jobBean.getJobName(),jobBean.getCronExpression(),null);
        }
    }
}

对于怎么做到动态的添加、暂停、删除、恢复、执行一个任务呢? quartz提供了这样的API。我们只需要稍微做一下封装。以接口的形式提供出去就可以了。

package com.xchaset.quartz.schedule;

import org.quartz.*;
import org.quartz.impl.matchers.GroupMatcher;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.scheduling.quartz.QuartzJobBean;
import org.springframework.stereotype.Service;

import javax.annotation.PostConstruct;
import java.util.*;

@Service
public class QuartzService {

    @Autowired
    private Scheduler scheduler;

    @PostConstruct
    public void startScheduler() {
        try {
            scheduler.start();
        } catch (SchedulerException e) {
            e.printStackTrace();
        }
    }

    /**
     * 添加一个任务
     * @param jobClass
     * @param jobGroup
     * @param jobName
     * @param jobTime
     * @param jobMap
     * @throws SchedulerException
     */
    public void addJob(Class<? extends QuartzJobBean> jobClass, String jobGroup, String jobName, String jobTime, Map jobMap) throws SchedulerException {
        JobDetail build = JobBuilder.newJob(jobClass).withIdentity(jobName, jobGroup).build();
        if (jobMap != null && jobMap.size() > 0) {
            build.getJobDataMap().putAll(jobMap);
        }
        Trigger cronTrigger = TriggerBuilder.newTrigger().withIdentity(jobName, jobGroup)
                .startAt(DateBuilder.futureDate(1, DateBuilder.IntervalUnit.SECOND))
                .withSchedule(CronScheduleBuilder.cronSchedule(jobTime)).startNow().build();

        scheduler.scheduleJob(build, cronTrigger);

    }

    /**
     * 更新任务时间表达式
     * @param jobGroup
     * @param jobName
     * @param jobTime
     * @throws SchedulerException
     */
    public void updateJob(String jobGroup, String jobName, String jobTime) throws SchedulerException {
        TriggerKey triggerKey = TriggerKey.triggerKey(jobName, jobGroup);
        CronTrigger trigger = (CronTrigger)scheduler.getTrigger(triggerKey);
        trigger = trigger.getTriggerBuilder().withIdentity(triggerKey)
                .withSchedule(CronScheduleBuilder.cronSchedule(jobTime)).build();
        scheduler.rescheduleJob(triggerKey,trigger);
    }

    /**
     * 删除任务
     * @param jobGroup
     * @param jobName
     * @throws SchedulerException
     */
    public void deleteJob(String jobGroup,String jobName) throws SchedulerException {
        scheduler.deleteJob(new JobKey(jobName,jobGroup));
    }

    public void pauseJob(String jobGroup,String jobName) throws SchedulerException {
        scheduler.pauseJob(new JobKey(jobName,jobGroup));
    }

    /**
     * 恢复任务
     * @param jobGroup
     * @param jobName
     * @throws SchedulerException
     */
    public void resumeJob(String jobGroup,String jobName) throws SchedulerException {
        scheduler.resumeJob(new JobKey(jobName,jobGroup));
    }

    /**
     * 立刻执行一个任务
     * @param jobGroup
     * @param jobName
     * @throws SchedulerException
     */
    public void runAJobNow(String jobGroup,String jobName) throws SchedulerException {
        scheduler.triggerJob(new JobKey(jobName,jobGroup));
    }

    /**
     * 查询所有的任务
     * @return
     * @throws SchedulerException
     */
    public List<JobDetailsBean> queryAllJob() throws SchedulerException {
        GroupMatcher<JobKey> anyJobGroup = GroupMatcher.anyJobGroup();
        Set<JobKey> jobKeys = scheduler.getJobKeys(anyJobGroup);
        List<JobDetailsBean> result = getJobDetailsBeans(jobKeys);
        return result;
    }

    private List<JobDetailsBean> getJobDetailsBeans(Collection<JobKey> jobKeys) throws SchedulerException {
        List<JobDetailsBean> result = new ArrayList<>();
        for (JobKey jobKey : jobKeys) {
            List<? extends Trigger> triggersOfJob = scheduler.getTriggersOfJob(jobKey);
            for (Trigger trigger : triggersOfJob) {
                String group = jobKey.getGroup();
                String name = jobKey.getName();
                String className = jobKey.getClass().getName();
                String description = trigger.getDescription();
                JobDataMap jobDataMap = trigger.getJobDataMap();
                Date nextFireTime = trigger.getNextFireTime();
                Date previousFireTime = trigger.getPreviousFireTime();
                Trigger.TriggerState triggerState = scheduler.getTriggerState(TriggerKey.triggerKey(name, group));
                String cronExpression = "";
                if (trigger instanceof CronTrigger) {
                    cronExpression = ((CronTrigger) trigger).getCronExpression();
                }

                JobDetailsBean build = JobDetailsBean.builder().className(className).groupName(group).jobName(name).description(description)
                        .cronExpression(cronExpression).jobDataMap(jobDataMap).nextFireTime(nextFireTime).previousFireTime(previousFireTime)
                        .triggerState(triggerState.name()).build();
                result.add(build);
            }
        }
        return result;
    }

    /**
     * 查询正在执行的任务
     * @return
     * @throws SchedulerException
     */
    public List<JobDetailsBean> queryCurrentExecutingJob() throws SchedulerException {
        List<JobDetailsBean> result = new ArrayList<>();
        List<JobExecutionContext> currentlyExecutingJobs = scheduler.getCurrentlyExecutingJobs();
        for (JobExecutionContext currentlyExecutingJob : currentlyExecutingJobs) {
            Trigger trigger = currentlyExecutingJob.getTrigger();
            JobDataMap jobDataMap = trigger.getJobDataMap();
            JobKey jobKey = trigger.getJobKey();
            String group = jobKey.getGroup();
            String name = jobKey.getName();
            String className = jobKey.getClass().getName();
            Date nextFireTime = trigger.getNextFireTime();
            String description = trigger.getDescription();
            Date previousFireTime = trigger.getPreviousFireTime();
            Trigger.TriggerState triggerState = scheduler.getTriggerState(trigger.getKey());
            String cronExpression = "";
            if (trigger instanceof CronTrigger) {
                cronExpression = ((CronTrigger) trigger).getCronExpression();
            }
            JobDetailsBean build = JobDetailsBean.builder().className(className).groupName(group).jobName(name).description(description)
                    .cronExpression(cronExpression).jobDataMap(jobDataMap).nextFireTime(nextFireTime).previousFireTime(previousFireTime)
                    .triggerState(triggerState.name()).build();
            result.add(build);
        }
        return result;
    }
}

我们可以在上面代码的基础上,对外提供接口,这样就可以通过可视化的界面来动态的对任务进行调度了。

以上就是springboot集成quartz的简单使用了。

源码可以参考github仓库: https://github.com/xchaset/example

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,658评论 6 496
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,482评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,213评论 0 350
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,395评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,487评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,523评论 1 293
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,525评论 3 414
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,300评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,753评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,048评论 2 330
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,223评论 1 343
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,905评论 5 338
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,541评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,168评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,417评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,094评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,088评论 2 352

推荐阅读更多精彩内容