spring-boot-route(二十一)quartz實現動態定時任務

Quartz是一個定時任務的調度框架,涉及到的主要概念有如下幾個:java

Scheduler:調度器,全部的調度都由它控制,全部的任務都由它管理。mysql

Job:任務,定義業務邏輯。git

JobDetail:基於Job,進一步封裝。其中關聯一個Job,併爲Job指定更詳細的信息。github

Trigger:觸發器,能夠指定給某個任務,指定任務的觸發機制。spring

一 建立簡單任務

1.1 Quartz依賴

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

1.2 建立任務

任務建立須要實現Job接口,重寫execute(JobExecutionContext jobExecutionContext)方法,增長定時任務的業務邏輯,這裏我只是簡單的打印一下定時任務執行。sql

@Slf4j
public class SimpleJob implements Job {
    @Override
    public void execute(JobExecutionContext jobExecutionContext) throws JobExecutionException {
        log.info("job execute---"+new Date());
    }
}

1.3 JobDetail增長屬性

這裏增長的屬性能夠在Job實現類中獲取,來處理業務。數據庫

JobDetail jobDetail = JobBuilder.newJob(SimpleJob.class)
                                // 任務標識,及任務分組
                                .withIdentity("job1", "group1")
                                // 連接調用,增長鬚要的參數
                                .usingJobData("name","Java旅途")
                                .usingJobData("age",18)
                                .build();

1.4 Trigger實現

Trigger分爲兩種,SimpleTriggerCronTriggerSimpleTrigger是根據Quartz的一些api實現的簡單觸發行爲。CronTrigger用的比較多,使用cron表達式進行觸發。這裏先用SimpleTrigger來實現。api

SimpleTrigger simpleTrigger = TriggerBuilder.newTrigger()
                                            .withIdentity("trigger1", "group1")
                                            // 當即執行
                                            .startNow()
                                            // 10s後中止
                                            .endAt(new Date(System.currentTimeMillis()+10*1000))
                                            .withSchedule(
                                            SimpleScheduleBuilder.simpleSchedule()
                                            // 每秒執行一次
                                            .withIntervalInSeconds(1)
                                            // 一直執行
                                            .repeatForever()
                                            ).build();

1.5 啓動任務

@Autowired
private Scheduler scheduler;

scheduler.scheduleJob(jobDetail,simpleTrigger);

1.6 執行效果

啓動項目後,任務當即執行,每秒執行一次,10s後中止,執行效果圖以下:微信

二 動態操做定時任務

有時候除了已經開發好的定時任務外,還須要咱們手動去建立任務而且控制任務的執行。app

2.1 建立任務

@GetMapping("create")
public void createJob(String jobName,String jobGroup,String cron,String triggerName,String triggerGroup) throws SchedulerException {

    JobKey jobKey = new JobKey(jobName,jobGroup);
    // 若是存在這個任務,則刪除
    if(scheduler.checkExists(jobKey)) {
        scheduler.deleteJob(jobKey);
    }

    JobDetail jobDetail = JobBuilder.newJob(SimpleJob.class)
        .withIdentity(jobKey)
        .build();

    CronScheduleBuilder cronScheduleBuilder = CronScheduleBuilder.cronSchedule(cron);

    Trigger trigger = TriggerBuilder.newTrigger()
        .withIdentity(triggerName,triggerGroup)
        .withSchedule(cronScheduleBuilder).build();
    scheduler.scheduleJob(jobDetail,trigger);
}

2.2 暫停任務

@GetMapping("pause")
public void pauseJob(String jobName,String jobGroup) throws SchedulerException {

    JobKey jobKey = JobKey.jobKey(jobName, jobGroup);
    JobDetail jobDetail = scheduler.getJobDetail(jobKey);
    if (jobDetail == null) {
        return;
    }
    scheduler.pauseJob(jobKey);
}

2.3 恢復暫停的任務

@GetMapping("remuse")
public void remuseJob(String jobName, String jobGroup) throws SchedulerException {
    JobKey jobKey = JobKey.jobKey(jobName, jobGroup);
    JobDetail jobDetail = scheduler.getJobDetail(jobKey);
    if (jobDetail == null) {
        return;
    }
    scheduler.resumeJob(jobKey);
}

2.4 刪除定時任務

@GetMapping("remove")
public void removeJob(String jobName, String jobGroup,String triggerName,String triggerGroup) throws SchedulerException {

    TriggerKey triggerKey = TriggerKey.triggerKey(triggerName, triggerGroup);
    JobKey jobKey = JobKey.jobKey(jobName, jobGroup);
    Trigger trigger =  scheduler.getTrigger(triggerKey);
    if (trigger == null) {
        return;
    }
    // 中止觸發器
    scheduler.pauseTrigger(triggerKey);
    // 移除觸發器
    scheduler.unscheduleJob(triggerKey);
    // 刪除任務
    scheduler.deleteJob(jobKey);
}

三 任務持久化

Quartz默認使用RAMJobStore存儲方式將任務存儲在內存中,除了這種方式還支持使用JDBC將任務存儲在數據庫,爲了防止任務丟失,咱們通常會將任務存儲在數據庫中。

這裏使用mysql進行存儲,在quartz的源碼包中找到文件tables_mysql_innodb.sql,而後在客戶端進行運行sql文件。若是嫌源碼包很差下載的話,我已經將sql文件上傳至GitHub了,能夠直接訪問github拉去表結構,數據表以下:

3.1 增長mysql和jdbc依賴

<dependency>
    <groupId>mysql</groupId>
    <artifactId>mysql-connector-java</artifactId>
    <version>8.0.11</version>
</dependency>

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

3.2 指定使用jdbc存儲

quartz默認使用memory存儲,這裏修改爲jdbc進行存儲,並配置jdbc的相關信息

spring:
  quartz:
    job-store-type: jdbc
  datasource:
    driver-class-name: com.mysql.cj.jdbc.Driver
    url: jdbc:mysql://localhost:3306/simple_fast
    username: root
    password: root

3.3 建立任務

啓動項目,調用create接口建立任務,而後數據表中就會新增任務相關的數據了。

此是spring-boot-route系列的第二十一篇文章,這個系列的文章都比較簡單,主要目的就是爲了幫助初次接觸Spring Boot 的同窗有一個系統的認識。本文已收錄至個人github,歡迎各位小夥伴star

githubhttps://github.com/binzh303/s...

點關注、不迷路

若是以爲文章不錯,歡迎關注點贊收藏,大家的支持是我創做的動力,感謝你們。

若是文章寫的有問題,請不要吝惜文筆,歡迎留言指出,我會及時覈查修改。

若是你還想更加深刻的瞭解我,能夠微信搜索「Java旅途」進行關注。回覆「1024」便可得到學習視頻及精美電子書。天天7:30準時推送技術文章,讓你的上班路不在孤獨,並且每個月還有送書活動,助你提高硬實力!

相關文章
相關標籤/搜索