Quartz任务调度

juio 9年前

 注意:

 不同的版本的jar包,具体的操作不太相同,但是思路是相同的;比如1.8.6jar包中,JobDetail是个类,直接通过构造方法与Job类关联。SimpleTrigger和CornTrigger是类;在2.2.1jar包中,JobDetail是个接口,SimpleTrigger和CornTrigger是接口

mavan工程引入的包:

<!-- 任务调度器相关包 -->   <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>

代码清单1 SimpleJob:简单的Job实现类:

import org.quartz.Job;  import org.quartz.JobExecutionContext;  import org.quartz.JobExecutionException;    public class SimpleJob implements Job {      /**       * @param context       * @throws JobExecutionException       */      @Override      public void execute(JobExecutionContext context)              throws JobExecutionException {          System.out.println("************哈哈哈********************");      }  }

这个类用一条非常简单的输出语句实现了Job接口的execute(JobExecutionContext context) 方法,这个方法可以包含想要执行的任何代码。下面,我们通过SimpleTrigger以及CronTrigger对SimpleJob进行调度:

代码清单2 SimpleTriggerRunner:使用SimpleTrigger进行调度

import static org.quartz.DateBuilder.evenMinuteDate;  import static org.quartz.JobBuilder.newJob;  import static org.quartz.TriggerBuilder.newTrigger;    import java.util.Date;    import org.quartz.CronScheduleBuilder;  import org.quartz.CronTrigger;  import org.quartz.JobDetail;  import org.quartz.Scheduler;  import org.quartz.SchedulerFactory;  import org.quartz.Trigger;  import org.quartz.TriggerBuilder;  import org.quartz.impl.StdSchedulerFactory;    public class SimpleTriggerRunner {      public void run() throws Exception {        // 通过SchedulerFactory获取一个调度器实例            SchedulerFactory sf = new StdSchedulerFactory();          Scheduler sched = sf.getScheduler();          //创建任务          JobDetail job = newJob(SimpleJob.class).withIdentity("job1", "group1")                  .build();          //创建触发器每天9点17分触发          CronTrigger trigger = TriggerBuilder.newTrigger()                  .withIdentity("trigger1", "group1")                  .withSchedule(CronScheduleBuilder.cronSchedule("0 17 9 * * ? *"))                  .build();          //注册调度任务          sched.scheduleJob(job, trigger);          /*          //在现在的时间上加一分钟          Date runTime = evenMinuteDate(new Date());          //创建简单的触发器一分钟后触发          Trigger trigger2 = newTrigger().withIdentity("trigger2", "group1")                  .startAt(runTime)                  .build();          sched.scheduleJob(job, trigger2);          */          //启动任务调度          sched.start();          try {              Thread.sleep(1000);          }          catch (Exception e) {            //调度器停止运行                sched.shutdown(true);            }        }            public static void main(String[] args) throws Exception {          SimpleTriggerRunner example = new SimpleTriggerRunner();          example.run();      }  }

Spring中使用任务调度器

一、 说明

     spring3.1以下的版本必须使用quartz1.x系列,3.1以上的版本才支持quartz 2.x,不然会出错。至于原因,则是spring对于quartz的支持实现,org.springframework.scheduling.quartz.CronTriggerBean继承了org.quartz.CronTrigger,在quartz1.x系列中org.quartz.CronTrigger是个类,而在quartz2.x系列中org.quartz.CronTrigger变成了接口,从而造成无法用spring的方式配置quartz的触发器(trigger)。

     我使用的quartz版本是2.2.1 。

     最终实现的功能:

      1) 项目启动时,可执行的定时任务启动,按时执行相应的逻辑 ;

     2)  可添加新任务,删除任务,更新任务,暂停任务,恢复任务 ;

二、 添加quartz包

    我使用Maven构建项目,包请见上文mavan中pom.xml

三、 配置及使用

      1.  配置任务调度器 (对应的文件名为quartz-task.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-3.0.xsd"   default-lazy-init="false">   <!-- 调度器 -->      <bean name="schedulerFactoryBean" class="org.springframework.scheduling.quartz.SchedulerFactoryBean">          <!-- 通过applicationContextSchedulerContextKey属性配置spring上下文 -->              <property name="applicationContextSchedulerContextKey">                  <value>applicationContext</value>              </property>         </bean>        <!--加载可执行的任务-->      <bean id="loadTask" class="com.quartz.LoadTask" init-method="initTask" />     </beans>

2. 服务器启动时加载,在web.xml文件里配置

<context-param>    <param-name>contextConfigLocation</param-name>    <param-value>classpath:quartz-task.xml</param-value>   </context-param>

 3. 加载可执行任务的类LoadTask.java

public class LoadTask {   public void initTask() throws Exception {    Scheduler scheduler = schedulerFactoryBean.getScheduler();    // 可执行的任务列表    Collection<Task> taskList = taskService.findTask();    for (Task task : taskList) {     // 任务名称和任务组设置规则:     // 名称:task_1 ..     // 组 :group_1 ..     TriggerKey triggerKey = TriggerKey.triggerKey(       "task_" + task.getId(), "group_" + task.getId());     CronTrigger trigger = (CronTrigger) scheduler       .getTrigger(triggerKey);     // 不存在,创建一个     if (null == trigger) {      JobDetail jobDetail = JobBuilder        .newJob(QuartzJobFactory.class)        .withIdentity("task_" + task.getId(),          "group_" + task.getId()).build();      jobDetail.getJobDataMap().put("scheduleJob", task);      // 表达式调度构建器      CronScheduleBuilder scheduleBuilder = CronScheduleBuilder        .cronSchedule(getCronExpression());      // 按新的表达式构建一个新的trigger      trigger = TriggerBuilder        .newTrigger()        .withIdentity("task_" + task.getId(),          "group_" + task.getId())        .withSchedule(scheduleBuilder).build();      scheduler.scheduleJob(jobDetail, trigger);     } else {      // trigger已存在,则更新相应的定时设置      CronScheduleBuilder scheduleBuilder = CronScheduleBuilder        .cronSchedule(taskService.getCronExpression());      // 按新的cronExpression表达式重新构建trigger      trigger = trigger.getTriggerBuilder().withIdentity(triggerKey)        .withSchedule(scheduleBuilder).build();      // 按新的trigger重新设置job执行      scheduler.rescheduleJob(triggerKey, trigger);     }    }   }   @Autowired   private SchedulerFactoryBean schedulerFactoryBean;   @Autowired   private  TaskService taskService;  }

 4. 调度任务的入口

 public class QuartzTaskFactory implements Job {       @Override   public void execute(JobExecutionContext context)     throws JobExecutionException {    // TODO Auto-generated method stub    try {     System.out.println("任务运行...");     Task task = (Task) context.getMergedJobDataMap().get(       "scheduleJob");     System.out.println("任务名称: [" + task.getTaskName() + "]");     //在这里执行你的任务...     } catch (Exception e) {     e.printStackTrace();    }   }  }

 5. 暂停任务

Scheduler scheduler = schedulerFactoryBean.getScheduler();  JobKey jobKey = JobKey.jobKey(scheduleJob.getJobName(), scheduleJob.getJobGroup());  scheduler.pauseJob(jobKey);

6. 恢复任务

Scheduler scheduler = schedulerFactoryBean.getScheduler();  JobKey jobKey = JobKey.jobKey(scheduleJob.getJobName(), scheduleJob.getJobGroup());  scheduler.resumeJob(jobKey);

 7. 删除任务

Scheduler scheduler = schedulerFactoryBean.getScheduler();  JobKey jobKey = JobKey.jobKey(scheduleJob.getJobName(), scheduleJob.getJobGroup());  scheduler.deleteJob(jobKey);

8. 立即运行任务

Scheduler scheduler = schedulerFactoryBean.getScheduler();  JobKey jobKey = JobKey.jobKey(scheduleJob.getJobName(), scheduleJob.getJobGroup());  scheduler.triggerJob(jobKey);

9. 更新任务(时间表达式)

Scheduler scheduler = schedulerFactoryBean.getScheduler();      TriggerKey triggerKey = TriggerKey.triggerKey(scheduleJob.getJobName(),  scheduleJob.getJobGroup());      //获取trigger,即在spring配置文件中定义的 bean id="myTrigger"  CronTrigger trigger = (CronTrigger) scheduler.getTrigger(triggerKey);      //表达式调度构建器  CronScheduleBuilder scheduleBuilder = CronScheduleBuilder.cronSchedule(scheduleJob  .getCronExpression());      //按新的cronExpression表达式重新构建trigger  trigger = trigger.getTriggerBuilder().withIdentity(triggerKey)  .withSchedule(scheduleBuilder).build();      //按新的trigger重新设置job执行  scheduler.rescheduleJob(triggerKey, trigger);

四、时间表达式说明

字段 允许值 允许的特殊字符

秒 0-59 , – * /

分 0-59 , – * /

小时 0-23 , – * /

日期 1-31 , – * ? / L W C

月份 1-12 或者 JAN-DEC , – * /

星期 1-7 或者 SUN-SAT , – * ? / L C #

年(可选) 留空, 1970-2099 , – * /

表达式意义

"0 0 12 * * ?" 每天中午12点触发

"0 15 10 ? * *" 每天上午10:15触发

"0 15 10 * * ?" 每天上午10:15触发

"0 15 10 * * ? *" 每天上午10:15触发

"0 15 10 * * ? 2005" 2005年的每天上午10:15触发

"0 * 14 * * ?" 在每天下午2点到下午2:59期间的每1分钟触发

"0 0/5 14 * * ?" 在每天下午2点到下午2:55期间的每5分钟触发

"0 0/5 14,18 * * ?" 在每天下午2点到2:55期间和下午6点到6:55期间的每5分钟触发

"0 0-5 14 * * ?" 在每天下午2点到下午2:05期间的每1分钟触发

"0 10,44 14 ? 3 WED" 每年三月的星期三的下午2:10和2:44触发

"0 15 10 ? * MON-FRI" 周一至周五的上午10:15触发

"0 15 10 15 * ?" 每月15日上午10:15触发

"0 15 10 L * ?" 每月最后一日的上午10:15触发

"0 15 10 ? * 6L" 每月的最后一个星期五上午10:15触发

"0 15 10 ? * 6L 2002-2005" 2002年至2005年的每月的最后一个星期五上午10:15触发

"0 15 10 ? * 6#3" 每月的第三个星期五上午10:15触发

每天早上6点

0 6 * * *

每两个小时

0 */2 * * *

晚上11点到早上8点之间每两个小时,早上八点

0 23-7/2,8 * * *

每个月的4号和每个礼拜的礼拜一到礼拜三的早上11点

0 11 4 * 1-3

1月1日早上4点

0 4 1 1 *

ok,定时任务已经正确执行....