使用quartz實現定時功能

Quartz是一個完全由java編寫的開源作業調度框架,具體的介紹可到http://www.opensymphony.com/quartz/官方網站查看。
 
Quartz的幾個核心的接口和類爲:
 
Job接口:自己寫的“定時程序”實現此接口的void execute(JobExecutionContext arg0)方法,Job還有一類爲有狀態的StatefulJob接口,如果我們需要在上一個作業執行完後,根據其執行結果再進行下次作業的執行,則需要實現此接口。
Trigger抽象類:調度類(Scheduler)在時間到時調用此類,再由trigger類調用指定的定時程序。
Quertz中提供了兩類觸發器爲:SimpleTrigger,CronTrigger。前者用於實現比較簡單的定時功能,例如幾點開始,幾點結束,隔多長時間執行,共執行多少次等,後者提供了使用表達式來描述定時功能,因此適用於比較複雜的定時描述,例如每個月的最後一個週五,每週的週四等。
JobDetail類:具體某個定時程序的詳細描述,包括Name,Group,JobDataMap等。
JobExecutionContext類:定時程序執行的run-time的上下文環境,用於得到當前執行的Job的名字,配置的參數等。
JobDataMap類:用於描述一個作業的參數,參數可以爲任何基本類型例如String,float等,也可爲某個對象的引用.
JobListener,TriggerListener接口:用於監聽觸發器狀態和作業掃行狀態,在特寫狀態執行相應操作。
JobStore類:在哪裏執行定進程序,可選的有在內存中,在數據庫中。
 
簡單的定時程序:
 
public class TestJob implements Job
{
   public TestJob(){}
   public void execute(JobExecutionContext arg0) throws JobExecutionException
   {
      String name = context.getJobDetail().getJobDataMap().getString("name");
      System.out.println("job executing..."+name);   }
}
 
public class QuartzTest
{
 public static void main(String[] args)
 {
    QuartzTest test = new QuartzTest();
    try
   {
      test.startSchedule();
   }
  catch (Exception e)
  {
     e.printStackTrace();
  }
 }
 public void startSchedule() throws Exception
 {
     Scheduler scheduler = StdSchedulerFactory.getDefaultScheduler();
     JobDetail jobDetail =
      new JobDetail("testJob", Scheduler.DEFAULT_GROUP, TestJob.class);
      //結束時間     
     long end = System.currentTimeMillis() + 9000L;
    //執行10次,每3秒執行一次,到9秒後結束
     SimpleTrigger trigger = new SimpleTrigger("test",null,new Date(),new Date(end),10,3000L);
      scheduler.scheduleJob(jobDetail, trigger);
     scheduler.start();
 }
}
 
執行上面這個類基本實現了一個簡單的定時程序。但問題是現在這個類只能應用在application中,在web環境裏執行還需要添加一些配置,例如添加servlet,添加配置文件quartz.properties或者quartz-job.xml(在XML文件裏以配置方式定義triiger,定時描述等)。
 
web應用中使用
 

在web.xml中添加QuartzInitializerServlet,Quartz爲能夠在web應用中使用,提供了一個QuartzInitializerServlet和一個QuartzInitializerListener,用於在加載web應用時,對quartz進行初始化。我在使用servlet時加載成功,在使用listener時不成功,不知道怎麼回事?

servlet配置: 
<servlet> 
   <servlet-name>QuartzInitializer</servlet-name> 
   <servlet-class>org.quartz.ee.servlet.QuartzInitializerServlet</servlet-class> 
   <init-param>
     <param-name>shutdown-on-unload</param-name>
     <param-value>true</param-value>
   </init-param>
   <init-param>
    <param-name>config-file</param-name>
    <param-value>quartz.properties</param-value> 
   </init-param>
   <load-on-startup>2</load-on-startup>
 </servlet>
listener配置可以看源碼,主要的上面的參數配置爲<context-param>,再加一個<listener>.
 
上面提到了quartz.properties,這是自行指定的,Quartz提供了一個默認的配置文件,可以滿足基本的j2se應用,如果在web應用中,我們想把job,trigger配置都寫到文件中,就需要自己來寫,並指定在初始化時加載我們自己的quratz.properties,位置放在classes下。
 
#============================================================================
# Configure Main Scheduler Properties  
#============================================================================
org.quartz.scheduler.instanceName = org.quartz.scheduler.instanceId = AUTO
#============================================================================
# Configure ThreadPool  
#============================================================================
org.quartz.threadPool.class = org.quartz.simpl.SimpleThreadPool
org.quartz.threadPool.threadCount = 3
org.quartz.threadPool.threadPriority = 5
#============================================================================
# Configure Plugins 
#============================================================================
org.quartz.plugin.triggHistory.class = org.quartz.plugins.history.LoggingJobHistoryPlugin
org.quartz.plugin.jobInitializer.class = org.quartz.plugins.xml.JobInitializationPlugin
org.quartz.plugin.jobInitializer.fileName = /scheduler/quartz_jobs.xml
org.quartz.plugin.jobInitializer.overWriteExistingJobs = true
org.quartz.plugin.jobInitializer.failOnFileNotFound = true
org.quartz.plugin.jobInitializer.scanInterval = 10
 
quartz要使用插件來加載自己的xml配置文件,上邊我們指定了在初始化時加載classes/scheduler/quartz_jobs.xml,默認的是加載classes/quartz_jobs.xml文件。
 
quartz_jobs.xml文件:

<?xml version='1.0' encoding='utf-8'?>
<quartz>
 <job>
   <job-detail>
    <name>test</name>
    <group>DEFAULT</group>
    <description>testJobhere</description>
    <job-class>TestJob</job-class>
    <job-data-map allows-transient-data="true">
     <entry>
      <key>name</key>
      <value>test</value>
     </entry>
   </job-data-map>
   </job-detail>
   <trigger>
             <cron>
                  <name>testCron</name>
                  <group>DEFAULT</group>
                  <job-name>test</job-name>
                 <job-group>DEFALUT</job-group>
                 <cron-expression>0/3 * * * * ?</cron-expression>
             </cron>
       </trigger>
  </job>
</quartz>

 
上面配置了一個作業,並聲明一個參數Name;配置了一個CronTrigger,每三秒執行一次。如果要配置SimpleTrigger ,可以使用<simple>標籤。
 
上面與Job對應的類爲TestJob,源碼爲:
public class TestJob implements Job
{
 public TestJob(){}
 public void execute(JobExecutionContext context) throws JobExecutionException
 {
     String name = context.getJobDetail().getJobDataMap().getString("name");
     System.out.println("job executing..."+name);
 }
}
 
在quartz_job.xml文件中還可以指定TriggerListener,JobListener等,可以使用<trigger-listener>,<job-listener>標籤來指定。
 
由於quartz目前文檔不是很多,大部分都是看源碼。總體看Quartz提供的CronTrigger使用表達式方式描述定時規律這個功能還是很強大的,在其源碼中有很多例子。
 
spring已經把quartz集成在一起,並進行了封裝,使用起來還是很方便的。
發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章