天天看點

使用Spring + quartz實作定時任務排程一.環境準備二.Spring配置三.代碼清單

需求如下:使用者可以設定定時發送短信.

一.環境準備

1.maven導入

<dependency>
        <groupId>org.quartz-scheduler</groupId>
        <artifactId>quartz</artifactId>
        <version>2.2.1</version>
    </dependency>
    <dependency>
        <groupId>org.quartz-scheduler</groupId>
        <artifactId>quartz-jobs</artifactId>
        <version>2.2.1</version>
    </dependency>
           

2.資料庫建立quartz表

網上找.注意替換

ENGINE=InnoDB

3.classpath下建立配置檔案:

quartz.properties

# Configure Main Scheduler Properties
org.quartz.scheduler.instanceName=MyTestQuartz
org.quartz.scheduler.instanceId=AUTO
org.quartz.scheduler.skipUpdateCheck=true

# Configure ThreadPool
org.quartz.threadPool.class=org.quartz.simpl.SimpleThreadPool 
org.quartz.threadPool.threadCount=
org.quartz.threadPool.threadPriority= 

# Configure JobStore 
#org.quartz.jobStore.class=org.quartz.simpl.RAMJobStore//儲存在記憶體
org.quartz.jobStore.class=org.quartz.impl.jdbcjobstore.JobStoreTX
org.quartz.jobStore.driverDelegateClass=org.quartz.impl.jdbcjobstore.StdJDBCDelegate
org.quartz.jobStore.dataSource=QRTZ_DS
//預設字首 可以按需修改
org.quartz.jobStore.tablePrefix=QRTZ_ 
org.quartz.jobStore.isClustered=true

#config datasource
org.quartz.dataSource.QRTZ_DS.driver=com.mysql.jdbc.Driver 
org.quartz.dataSource.QRTZ_DS.URL=jdbc:mysql://:/testdb?createDatabaseIfNotExist=true&autoReconnect=true&useUnicode=true&characterEncoding=utf8
org.quartz.dataSource.QRTZ_DS.user=root 
org.quartz.dataSource.QRTZ_DS.password=root
org.quartz.dataSource.QRTZ_DS.maxConnections=

# Configure Plugins
org.quartz.plugin.triggHistory.class=org.quartz.plugins.history.LoggingJobHistoryPlugin 
           

二.Spring配置

applicationContext.quartz.xml

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
    xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
    xsi:schemaLocation="http://www.springframework.org/schema/beans
            http://www.springframework.org/schema/beans/spring-beans.xsd">

    <bean name="schedulerFactoryBean" lazy-init="false"
        class="org.springframework.scheduling.quartz.SchedulerFactoryBean">
        <property name="configLocation" value="classpath:quartz.properties" />
    </bean>

    <bean id="quartzService"
        class="com.wlqq.mcs.admin.timer.quartz.service.impl.QuartzServiceImpl">
        <property name="sched" ref="schedulerFactoryBean" />
    </bean>

</beans>
           

三.代碼清單

目錄結構如下圖所示

使用Spring + quartz實作定時任務排程一.環境準備二.Spring配置三.代碼清單

1.QuartzService.java

package xxx;

import java.text.ParseException;
import java.util.Date;
import java.util.concurrent.TimeUnit;

import org.quartz.Job;
import org.quartz.JobDataMap;
import org.quartz.SchedulerException;

public interface QuartzService {

    /**
     * 添加按間隔執行任務
     * @param jobName 任務名稱
     * @param jobGroup 任務組名稱
     * @param triggerName 觸發器名稱
     * @param triggerGroup 觸發器組名稱
     * @param job 任務執行個體
     * @param newJobDataMap 狀态參數值
     * @param interval 時間間隔
     * @param timeUnit 時間機關
     * @return
     * @throws SchedulerException
     * @throws ParseException
     */
    Date addIntervalJob(String jobName, String jobGroup, String triggerName,
                        String triggerGroup, Job job, JobDataMap newJobDataMap,
                        Integer interval, TimeUnit timeUnit) throws SchedulerException, ParseException;

    /**
     * 添加按間隔執行任務
     * @param job 任務執行個體
     * @param newJobDataMap 狀态參數值
     * @param interval 時間間隔
     * @param timeUnit 時間機關
     * @return
     * @throws SchedulerException
     * @throws ParseException
     */
    Date addIntervalJob(Job job, JobDataMap newJobDataMap,
                        Integer interval, TimeUnit timeUnit) throws SchedulerException, ParseException;

    /**
     * 修改按間隔執行任務
     * @param jobName 任務名稱
     * @param jobGroup 任務組名稱
     * @param triggerName 觸發器名稱
     * @param triggerGroup 觸發器組名稱
     * @param job 任務執行個體
     * @param newJobDataMap 狀态參數值
     * @param interval 時間間隔
     * @param timeUnit 時間機關
     * @return
     * @throws SchedulerException
     * @throws ParseException
     */
    Date modifyIntervalJob(String jobName, String jobGroup, String triggerName,
                           String triggerGroup, Job job, JobDataMap newJobDataMap,
                           Integer interval, TimeUnit timeUnit) throws SchedulerException, ParseException;

    /**
     * 修改按間隔執行任務時間間隔
     * @param triggerName 觸發器名稱
     * @param triggerGroup 觸發器組名稱
     * @param interval 時間間隔
     * @param timeUnit 時間機關
     * @return
     * @throws SchedulerException
     * @throws ParseException
     */
    Date modifyJobInterval(String triggerName, String triggerGroup,
                           Integer interval, TimeUnit timeUnit) throws SchedulerException, ParseException;

    /**
     * 添加計劃任務
     * @param jobName 任務名稱
     * @param jobGroup 任務組名稱
     * @param triggerName 觸發器名稱
     * @param triggerGroup 觸發器組名稱
     * @param job 任務執行個體
     * @param newJobDataMap 狀态參數值
     * @param cronExpression cron執行表達式
     * @return
     * @throws SchedulerException
     * @throws ParseException
     */
    Date addCronJob(String jobName, String jobGroup, String triggerName,
                    String triggerGroup, Job job, JobDataMap newJobDataMap, String cronExpression)
            throws SchedulerException, ParseException;

    /**
     * 添加計劃任務
     * @param job 任務執行個體
     * @param newJobDataMap 狀态參數值
     * @param cronExpression cron執行表達式
     * @return
     * @throws SchedulerException
     * @throws ParseException
     */
    public Date addCronJob(Job job, JobDataMap newJobDataMap,String cronExpression)
            throws SchedulerException, ParseException;

    /**
     * 修改計劃任務
     * 
     * @param jobName
     * @param jobGroup
     * @param triggerName
     * @param triggerGroup
     * @param job
     * @param cronExpression
     * @return
     * @throws SchedulerException
     * @throws ParseException
     */
    Date modifyCronJob(String jobName, String jobGroup, String triggerName,
                       String triggerGroup, Job job, JobDataMap newJobDataMap, String cronExpression)
            throws SchedulerException, ParseException;

    /**
     * 修改計劃任務的觸發時間
     * 
     * @param triggerName
     * @param triggerGroupName
     * @param cronExpression
     * @throws SchedulerException
     * @throws ParseException
     */
    Date modifyCronJobTime(String triggerName, String triggerGroupName,
                           String cronExpression) throws SchedulerException, ParseException;

    /**
     * 移除任務
     * 
     * @param jobName
     * @param jobGroup
     * @param triggerName
     * @param triggerGroupName
     * @throws SchedulerException
     */
    boolean removeJob(String jobName, String jobGroup,
                      String triggerName, String triggerGroupName)
            throws SchedulerException;

    /**
     * 暫停任務
     * 
     * @param jobName
     * @param jobGroup
     * @return
     * @throws SchedulerException
     */
    void pauseJob(String jobName, String jobGroup,
                  String triggerName, String triggerGroupName)
            throws SchedulerException;


    /**
     * 恢複任務
     * 
     * @param jobName
     * @param jobGroup
     * @return
     * @throws SchedulerException
     */
    void resumeJob(String jobName, String jobGroup,
                   String triggerName, String triggerGroupName)
            throws SchedulerException;

}
           

2.QuartzServiceImpl.java

package xxx;

import static org.quartz.JobBuilder.newJob;
import static org.quartz.TriggerBuilder.newTrigger;
import java.text.ParseException;
import java.util.Date;
import java.util.UUID;
import java.util.concurrent.TimeUnit;
import org.quartz.CronScheduleBuilder;
import org.quartz.Job;
import org.quartz.JobBuilder;
import org.quartz.JobDataMap;
import org.quartz.JobDetail;
import org.quartz.JobKey;
import org.quartz.ScheduleBuilder;
import org.quartz.Scheduler;
import org.quartz.SchedulerException;
import org.quartz.SimpleScheduleBuilder;
import org.quartz.Trigger;
import org.quartz.TriggerKey;

import xxx.QuartzService;

public class QuartzServiceImpl implements QuartzService{
    private Scheduler sched;

    public void setSched(Scheduler sched) {
        this.sched = sched;
    }


    private Date addJob(String jobName, String jobGroup, String triggerName,
                        String triggerGroup, Job job, JobDataMap newJobDataMap,
                        ScheduleBuilder scheduleBuilder) throws SchedulerException, ParseException  {
        if (scheduleBuilder == null) {
            return null;
        }
        JobBuilder builder = newJob(job.getClass());
        builder.setJobData(newJobDataMap);
        JobDetail jobDetail = builder.withIdentity(jobName, jobGroup).build();
        // 觸發器
        Trigger trigger = newTrigger().withIdentity(triggerName, triggerGroup)
                .withSchedule(scheduleBuilder)
                .build();
        Date date = sched.scheduleJob(jobDetail, trigger);
        // 啟動
        if (!sched.isShutdown()) {
            sched.start();
        }
        return date;
    }

    private Date modifyJobScheduler(String triggerName, String triggerGroupName,
                                    ScheduleBuilder scheduleBuilder) throws SchedulerException, ParseException {
        if (scheduleBuilder == null) {
            return null;
        }
        TriggerKey key = new TriggerKey(triggerName, triggerGroupName);
        Trigger trigger = sched.getTrigger(key);
        if (trigger != null) {
            trigger.getTriggerBuilder()
                    .withSchedule(scheduleBuilder)
                    .build();
            return sched.rescheduleJob(key, trigger);
        }
        return null;
    }

    private SimpleScheduleBuilder intervalScheduleBuilder(Integer interval, TimeUnit timeUnit) {
        SimpleScheduleBuilder simpleScheduleBuilder = SimpleScheduleBuilder.simpleSchedule().repeatForever();

        switch (timeUnit) {
            case MINUTES:
                simpleScheduleBuilder.withIntervalInMinutes(interval);
                break;
            case HOURS:
                simpleScheduleBuilder.withIntervalInHours(interval);
                break;
            case DAYS:
                simpleScheduleBuilder.withIntervalInHours(interval * );
                break;
            default:
                return null;
        }
        return simpleScheduleBuilder;
    }

    @Override
    public Date addIntervalJob(String jobName, String jobGroup, String triggerName, String triggerGroup,
                               Job job, JobDataMap newJobDataMap,
                               Integer interval, TimeUnit timeUnit) throws SchedulerException, ParseException {

        return addJob(jobName, jobGroup, triggerName, triggerGroup, job, newJobDataMap,
                intervalScheduleBuilder(interval, timeUnit));
    }

    @Override
    public Date addIntervalJob(Job job, JobDataMap newJobDataMap,
                               Integer interval, TimeUnit timeUnit) throws SchedulerException, ParseException {
        String id=UUID.randomUUID().toString().replaceAll("-", "");
        return addIntervalJob(id, id, id, id, job, newJobDataMap, interval, timeUnit);
    }

    @Override
    public Date modifyIntervalJob(String jobName, String jobGroup, String triggerName, String triggerGroup,
                                  Job job, JobDataMap newJobDataMap,
                                  Integer interval, TimeUnit timeUnit) throws SchedulerException, ParseException {
        removeJob(jobName, jobGroup, triggerName, triggerGroup);
        return addIntervalJob(jobName, jobGroup, triggerName, triggerGroup, job, newJobDataMap, interval, timeUnit);
    }

    @Override
    public Date modifyJobInterval(String triggerName, String triggerGroup,
                                  Integer interval, TimeUnit timeUnit) throws SchedulerException, ParseException {
        return modifyJobScheduler(triggerName, triggerGroup, intervalScheduleBuilder(interval, timeUnit));
    }


    /**
     * 添加任務
     * 
     * @param jobName
     *            任務名稱
     * @param jobGroup
     *            任務組名稱
     * @param triggerName
     *            觸發器名稱
     * @param triggerGroup
     *            觸發器組名稱
     * @param job
     *            任務執行個體
     * @param cronExpression
     *            cron表達式
     * @throws SchedulerException
     * @throws ParseException
     */
    public Date addCronJob(String jobName, String jobGroup, String triggerName,
            String triggerGroup, Job job, JobDataMap newJobDataMap,
            String cronExpression) throws SchedulerException, ParseException {
        CronScheduleBuilder cronScheduleBuilder = CronScheduleBuilder.cronSchedule(cronExpression);
        return addJob(jobName, jobGroup, triggerName, triggerGroup, job, newJobDataMap, cronScheduleBuilder);
    }


    @Override
    public Date addCronJob(Job job, JobDataMap newJobDataMap, String cronExpression)
            throws SchedulerException, ParseException {
        String id=UUID.randomUUID().toString().replaceAll("-", "");
        return addCronJob(id, id, id, id, job, newJobDataMap, cronExpression);
    }

    /**
     * 修改任務
     * 
     * @param jobName
     * @param jobGroup
     * @param triggerName
     * @param triggerGroup
     * @param job
     * @param cronExpression
     * @return
     * @throws SchedulerException
     * @throws ParseException
     */
    public Date modifyCronJob(String jobName, String jobGroup, String triggerName,
            String triggerGroup, Job job,JobDataMap newJobDataMap, String cronExpression)
            throws SchedulerException, ParseException {
        removeJob(jobName, jobGroup, triggerName, triggerGroup);
        return addCronJob(jobName, jobGroup, triggerName, triggerGroup, job, newJobDataMap,
                cronExpression);
    }

    /**
     * 修改一個任務的觸發時間
     * 
     * @param triggerName
     * @param triggerGroupName
     * @param cronExpression
     * @throws SchedulerException
     * @throws ParseException
     */
    public Date modifyCronJobTime(String triggerName, String triggerGroupName,
            String cronExpression) throws SchedulerException, ParseException {
        return modifyJobScheduler(triggerName, triggerGroupName,
                CronScheduleBuilder.cronSchedule(cronExpression));
    }

    /**
     * 移除任務
     * 
     * @param jobName
     * @param jobGroup
     * @param triggerName
     * @param triggerGroupName
     * @throws SchedulerException
     */
    public boolean removeJob(String jobName, String jobGroup,
            String triggerName, String triggerGroupName)
            throws SchedulerException {
        TriggerKey key = new TriggerKey(triggerName, triggerGroupName);
        sched.pauseTrigger(key);// 停止觸發器
        sched.unscheduleJob(key);// 移除觸發器

        JobKey jobKey = new JobKey(jobName, jobGroup);
        return sched.deleteJob(jobKey);// 删除任務
    }

    /**
     * 暫停任務
     * 
     * @param jobName
     * @param jobGroup
     * @param triggerName
     * @param triggerGroupName
     * @throws SchedulerException
     */
    public void pauseJob(String jobName, String jobGroup, String triggerName,
            String triggerGroupName) throws SchedulerException {
        TriggerKey key = new TriggerKey(triggerName, triggerGroupName);
        sched.pauseTrigger(key);// 停止觸發器

        JobKey jobKey = new JobKey(jobName, jobGroup);
        sched.pauseJob(jobKey);// 停止任務
    }

    /**
     * 恢複任務
     * 
     * @param jobName
     * @param jobGroup
     * @param triggerName
     * @param triggerGroupName
     * @throws SchedulerException
     */
    public void resumeJob(String jobName, String jobGroup, String triggerName,
            String triggerGroupName) throws SchedulerException {
        TriggerKey key = new TriggerKey(triggerName, triggerGroupName);
        sched.resumeTrigger(key);// 重新開機觸發器

        JobKey jobKey = new JobKey(jobName, jobGroup);
        sched.resumeJob(jobKey);// 重新開機任務
    }
}
           

3.MassSmsJob.java (群發短信job)

package xxx.quartz.job;

import org.quartz.Job;
import org.quartz.JobDataMap;
import org.quartz.JobExecutionContext;
import org.quartz.JobExecutionException;
import org.springframework.beans.factory.annotation.Autowired;

import xxx.commons.exceptions.BusinessException;
import xxx.httpcommons.util.SpringContextUtil;
import xxx.service.SmsMassService;

/**
 * 群發短信job
 * @author Mingchenchen
 *
 */
public class MassSmsJob implements Job{
    public static final String SMS_MASS_JOB_GROUP = "smsMassJob"; // jobGroup
    public static final String SMS_MASS_TRIGGER_GROUP = "smsMassTrigger"; // triggerGroup

    @Override
    public void execute(JobExecutionContext context) throws JobExecutionException {
        JobDataMap datamap = context.getJobDetail().getJobDataMap();

        String smsMassUuid = datamap.getString("uuid");
        String title = datamap.getString("title");
        String message = datamap.getString("message");
        String phones = datamap.getString("phones");

        try {
            //此處就是定時任務實際執行的邏輯
            SmsMassService smsMassService = SpringContextUtil.getBean(SmsMassService.class);
            smsMassService.sendSms(smsMassUuid, title, message, phones);
        } catch (BusinessException e) {
            e.printStackTrace();
        }
    }

}

//業務邏輯 
public void sendSms(String smsMassUuid, String title,
        String message, String phones){
    //調用sms群發短信
    //TODO

    //修改資料庫标志位為已經發送
    smsMassDao.setSended(smsMassUuid);
}
           

4.CronExpressionUtils.java

package xxx.quartz.utils;

import java.util.Calendar;
import java.util.Date;

public class CronExpressionUtils {

    /**
     * 根據一個時間解析成cron表達式
     * @param date
     * @return
     */
    public static String getCronExpressionByDate(Date date){
        String cronExpression = "";

        Calendar calendar = Calendar.getInstance();
        calendar.setTime(date);
        int year = calendar.get(Calendar.YEAR);
        int month = calendar.get(Calendar.MONTH) + ; //注意從0開始
        int day = calendar.get(Calendar.DAY_OF_MONTH);
        int hour = calendar.get(Calendar.HOUR_OF_DAY);
        int minute = calendar.get(Calendar.MINUTE);
        int second = calendar.get(Calendar.SECOND);

        // "30 10 1 20 10 ? 2011" 2011年10月20号1點10分30秒觸發任務,?是第幾周 不指定
        cronExpression = second + " " + minute + " " + hour + " "
                + day + " " + month + " ? " + year;
        return cronExpression;
    }
}
           

5.業務邏輯service調用 SmsMassServiceImpl.java

@Value("${sms.mass.interval.minite:10}")
    private int smsMassInterval; //群發短信的間隔,預設是10分鐘
    @Autowired
    private SmsMassDao smsMassDao;
    @Autowired
    private QuartzService quartzService;

    /**
     * 建立定時任務
     * @param uuid
     * @param title
     * @param message
     * @param sendTime
     * @param phones
     * @throws BusinessException
     */
    private void createSmsMassCronJob(SmsMassRecord record) throws BusinessException{
        MassSmsJob job = new MassSmsJob();
        JobDataMap dataMap = new JobDataMap();
        dataMap.put("uuid", record.getUuid());
        dataMap.put("title", record.getTitle());
        dataMap.put("message", record.getMessage());
        dataMap.put("phones", record.getPhones());

        String jobName = record.getUuid();
        String jobGroup = MassSmsJob.SMS_MASS_JOB_GROUP;
        String triggerName = record.getUuid();
        String triggerGroup = MassSmsJob.SMS_MASS_TRIGGER_GROUP;

        Date date = record.getSendTime();
        String cronExpression = CronExpressionUtils.getCronExpressionByDate(date);
        try {
            quartzService.addCronJob(jobName, jobGroup, triggerName, 
                    triggerGroup, job, dataMap, cronExpression);
        } catch (SchedulerException | ParseException e) {
            e.printStackTrace();
            throw new BusinessException(ResultCode.SYSTEM_EXCEPTION.code(), ResultCode.SYSTEM_EXCEPTION.message());
        }
    }
           

其他地方需要調用SpringContextUtil.getBean(QuartzService.class);即可