天天看點

Java任務排程架構Quartz

 介紹

 Quartz is a full-featured, open source job scheduling service that can be integrated with, or used along side virtually any Java application - from the smallest stand-alone application to the largest e-commerce system. Quartz can be used to create simple or complex schedules for executing tens, hundreds, or even tens-of-thousands of jobs; 

 Quartz架構是一個全功能、開源的任務排程服務,可以內建幾乎任何的java應用程式—從小的單片機系統到大型的電子商務系統。Quartz可以執行上千上萬的任務排程。

 核心概念

 Quartz核心的概念:scheduler任務排程、Job任務、Trigger觸發器、JobDetail任務細節

 Job任務:其實Job是接口,其中隻有一個execute方法:

 package org.quartz;

public abstract interface Job

{

  public abstract void execute(JobExecutionContext paramJobExecutionContext)

    throws JobExecutionException;

}

 我們開發者隻要實作此接口,實作execute方法即可。把我們想做的事情,在execute中執行即可。

 JobDetail:任務細節,Quartz執行Job時,需要建立個Job執行個體,但是不能直接操作Job類,是以通過JobDetail來擷取Job的名稱、描述資訊。

 Trigger觸發器:執行任務的規則;比如每天,每小時等。

 一般情況使用SimpleTrigger,和CronTrigger,這個觸發器實作了Trigger接口。

 對于複雜的時間表達式來說,比如每個月15日上午幾點幾分,使用CronTrigger

 對于簡單的時間來說,比如每天執行幾次,使用SimpleTrigger

 scheduler任務排程:是最核心的概念,需要把JobDetail和Trigger注冊到scheduler中,才可以執行。

 具體執行步驟:

 下載下傳相應的jar包:​​http://www.quartz-scheduler.org/​​

 注意:

 不同的版本的jar包,具體的操作不太相同,但是思路是相同的;比如1.8.6jar包中,JobDetail是個類,直接通過構造方法與Job類關聯。SimpleTrigger和CornTrigger是類;在2.0.2jar包中,JobDetail是個接口,SimpleTrigger和CornTrigger是接口

 不同版本測試:

 1.8.6jar包:  

[html] view plain copy

 print?

package com.test;  

import java.util.Date;  

import org.quartz.Job;  

import org.quartz.JobExecutionContext;  

import org.quartz.JobExecutionException;  

/**  

 * 需要執行的任務  

 * @author lhy  

 *  

 */  

public class MyJob implements Job {  

    @Override  

    //把要執行的操作,寫在execute方法中  

    public void execute(JobExecutionContext arg0) throws JobExecutionException {  

        System.out.println("測試Quartz"+new Date());  

    }  

}  

 使用SimpleTrigger觸發器

import org.quartz.JobDetail;  

import org.quartz.Scheduler;  

import org.quartz.SchedulerException;  

import org.quartz.SchedulerFactory;  

import org.quartz.SimpleTrigger;  

import org.quartz.impl.StdSchedulerFactory;  

 * 調用任務的類  

public class SchedulerTest {  

   public static void main(String[] args) {  

     //通過schedulerFactory擷取一個排程器  

       SchedulerFactory schedulerfactory=new StdSchedulerFactory();  

       Scheduler scheduler=null;  

       try{  

//      通過schedulerFactory擷取一個排程器  

           scheduler=schedulerfactory.getScheduler();  

//       建立jobDetail執行個體,綁定Job實作類  

//       指明job的名稱,所在組的名稱,以及綁定job類  

           JobDetail jobDetail=new JobDetail("job1", "jgroup1", MyJob.class);  

//       定義排程觸發規則,比如每1秒運作一次,共運作8次  

           SimpleTrigger simpleTrigger=new SimpleTrigger("simpleTrigger","triggerGroup");  

//       馬上啟動  

           simpleTrigger.setStartTime(new Date());  

//       間隔時間  

           simpleTrigger.setRepeatInterval(1000);  

//       運作次數  

           simpleTrigger.setRepeatCount(8);  

//       把作業和觸發器注冊到任務排程中  

           scheduler.scheduleJob(jobDetail, simpleTrigger);  

//       啟動排程  

           scheduler.start();  

       }catch(SchedulerException e){  

           e.printStackTrace();  

       }  

 若使用CornTrigger觸發器: 

import org.quartz.CronTrigger;  

public class CronTriggerTest {  

//       定義排程觸發規則,每天上午10:15執行  

           CronTrigger cornTrigger=new CronTrigger("cronTrigger","triggerGroup");  

//       執行規則表達式  

           cornTrigger.setCronExpression("0 15 10 * * ? *");  

           scheduler.scheduleJob(jobDetail, cornTrigger);  

       }catch(Exception e){  

  對于2.0.2jar包如下:

  其中的job類不變,主要是排程類如下:

import org.quartz.CronScheduleBuilder;  

import org.quartz.JobBuilder;  

import org.quartz.SimpleScheduleBuilder;  

import org.quartz.Trigger;  

import org.quartz.TriggerBuilder;  

           JobDetail job=JobBuilder.newJob(MyJob.class).withIdentity("job1", "jgroup1").build();  

//       定義排程觸發規則  

//      使用simpleTrigger規則  

//        Trigger trigger=TriggerBuilder.newTrigger().withIdentity("simpleTrigger", "triggerGroup")  

//                        .withSchedule(SimpleScheduleBuilder.repeatSecondlyForever(1).withRepeatCount(8))  

//                        .startNow().build();  

//      使用cornTrigger規則  每天10點42分  

              Trigger trigger=TriggerBuilder.newTrigger().withIdentity("simpleTrigger", "triggerGroup")  

              .withSchedule(CronScheduleBuilder.cronSchedule("0 42 10 * * ? *"))  

              .startNow().build();   

           scheduler.scheduleJob(job, trigger);  

  上述demo下載下傳:1.8版本demo下載下傳

                2.0版本demo下載下傳

  對于CornExpress講解如下: 

字段   允許值   允許的特殊字元    

秒    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觸發     

特殊字元   意義    

*    表示所有值;    

?    表示未說明的值,即不關心它為何值;    

-    表示一個指定的範圍;    

,    表示附加一個可能值;    

/    符号前表示開始時間,符号後表示每次遞增的值;    

L("last")    ("last") "L" 用在day-of-month字段意思是 "這個月最後一天";用在 day-of-week字段, 它簡單意思是 "7" or "SAT"。 如果在day-of-week字段裡和數字聯合使用,它的意思就是 "這個月的最後一個星期幾" – 例如: "6L" means "這個月的最後一個星期五". 當我們用“L”時,不指明一個清單值或者範圍是很重要的,不然的話,我們會得到一些意想不到的結果。    

W("weekday")    隻能用在day-of-month字段。用來描叙最接近指定天的工作日(周一到周五)。例如:在day-of-month字段用“15W”指“最接近這個月第15天的工作日”,即如果這個月第15天是周六,那麼觸發器将會在這個月第14天即周五觸發;如果這個月第15天是周日,那麼觸發器将會在這個月第16天即周一觸發;如果這個月第15天是周二,那麼就在觸發器這天觸發。注意一點:這個用法隻會在目前月計算值,不會越過目前月。“W”字元僅能在day-of-month指明一天,不能是一個範圍或清單。也可以用“LW”來指定這個月的最後一個工作日。     

#    隻能用在day-of-week字段。用來指定這個月的第幾個周幾。例:在day-of-week字段用"6#3"指這個月第3個周五(6指周五,3指第3個)。如果指定的日期不存在,觸發器就不會觸發。     

C    指和calendar聯系後計算過的值。例:在day-of-month 字段用“5C”指在這個月第5天或之後包括calendar的第一天;在day-of-week字段用“1C”指在這周日或之後包括calendar的第一天  

作者:習慣沉澱