1. 程式人生 > >Spring Quartz 動態配置定時任務

Spring Quartz 動態配置定時任務

1、Quartz在Spring中的簡單配置
Spring配置檔案quartz.xml:
Java程式碼  收藏程式碼
  1. <?xml version="1.0" encoding="UTF-8"?>  
  2.         <!DOCTYPE beans PUBLIC "-//SPRING//DTD BEAN//EN""http://www.springframework.org/dtd/spring-beans.dtd">  
  3. <beans>  
  4.     <bean id="scheduleInfoService"class="com.erry.tntops.web.task.ScheduleInfoService"
    >  
  5.          <property name="scheduler" ref="schedulerFactory"/>  
  6.      </bean>  
  7.      <bean id="jobDetail"class="org.springframework.scheduling.quartz.MethodInvokingJobDetailFactoryBean">  
  8.          <property name="targetObject" ref="scheduleInfoService"/>  
  9.          <property name="targetMethod"
     value="test"/>  
  10.          <property name="concurrent" value="false"/>  
  11.      </bean>  
  12.      <bean id="cronTrigger"class="org.springframework.scheduling.quartz.CronTriggerBean" >  
  13.           <property name="jobDetail" ref="jobDetail"/>  
  14.           <property name="cronExpression"
    >  
  15.               <value>0/10 * * * * ?</value>  
  16.           </property>  
  17.       </bean>  
  18.      <bean id="schedulerFactory"class="org.springframework.scheduling.quartz.SchedulerFactoryBean">  
  19.          <property name="triggers">  
  20.              <list>  
  21.                  <ref local="cronTrigger"/>  
  22.              </list>  
  23.          </property>  
  24.     </bean>  
  25. </beans>  

在上面的配置中設定:
① targetMethod: 指定需要定時執行scheduleInfoAction中的test()方法
② concurrent:對於相同的JobDetail,當指定多個Trigger時, 很可能第一個job完成之前,第二個job就開始了。指定concurrent設為false,多個job不會併發執行,第二個job將不會在第一個job完成之前開始。
③ cronExpression:0/10 * * * * ?表示每10秒執行一次,具體可參考附表。
④ triggers:通過再新增其他的ref元素可在list中放置多個觸發器。
scheduleInfoAction中的simpleJobTest()方法
注意:此方法沒有引數,如果scheduleInfoAction有兩個方法test()和test(String argument),則spring只會去執行無參的test().
Java程式碼  收藏程式碼
  1. publicvoid test() {  
  2.          log.warn("uh oh, Job is scheduled !'" + "' Success...");  
  3. }  

2、Quartz在Spring中動態設定cronTrigger方法一
(1)、Spring配置檔案quartz.xml:
Java程式碼  收藏程式碼
  1. <?xml version="1.0" encoding="UTF-8"?>  
  2.         <!DOCTYPE beans PUBLIC "-//SPRING//DTD BEAN//EN""http://www.springframework.org/dtd/spring-beans.dtd">  
  3. <beans>  
  4.     <bean id="scheduleInfoAction"class="com.erry.tntops.web.task.ScheduleInfoAction">  
  5.         <property name="scheduler" ref="schedulerFactory"/>  
  6.         <!-- ref中的emsService是xml中配置的bean的id -->  
  7.         <property name="emsService" ref="emsService"/>  
  8.      </bean>  
  9.      <bean id="jobDetail"class="org.springframework.scheduling.quartz.MethodInvokingJobDetailFactoryBean">  
  10.          <property name="targetObject" ref="scheduleInfoAction"/>  
  11.          <property name="targetMethod" value="reScheduleJob"/>  
  12.          <property name="concurrent" value="false"/>  
  13.      </bean>  
  14.      <bean id="cronTrigger"class="com.erry.tntops.web.task.InitCronTrigger">  
  15.           <property name="jobDetail" ref="jobDetail"/>  
  16.           <property name="cronExpression">  
  17.               <value>0/10 * * * * ?</value>  
  18.           </property>  
  19.       </bean>  
  20.      <bean id="schedulerFactory"class="org.springframework.scheduling.quartz.SchedulerFactoryBean">  
  21.          <property name="triggers">  
  22.              <list>  
  23.                  <ref local="cronTrigger"/>  
  24.              </list>  
  25.          </property>  
  26.     </bean>  
  27. </beans>  

(2)、類ScheduleInfoAction:
Java程式碼  收藏程式碼
  1. import org.apache.log4j.Logger;  
  2. import org.quartz.Scheduler;  
  3. import org.quartz.SchedulerException;  
  4. import org.springframework.scheduling.quartz.CronTriggerBean;  
  5. import java.text.ParseException;  
  6. import java.util.Date;  
  7. publicclass ScheduleInfoAction{  
  8.     Logger logger = Logger.getLogger(ScheduleInfoAction.class);  
  9.      private Scheduler scheduler;  
  10.      // 設值注入,通過setter方法傳入被呼叫者的例項scheduler
  11.      publicvoid setScheduler(Scheduler scheduler) {  
  12.          this.scheduler = scheduler;  
  13.     }  
  14.     private EmsService emsService;  
  15.     publicvoid setEmsService(EmsService emsService){  
  16.         this.emsService = emsService;  
  17.     }  
  18.     publicvoid reScheduleJob() throws SchedulerException {  
  19.         // 執行時可通過動態注入的scheduler得到trigger,注意採用這種注入方式在有的專案中會有問題,如果遇到注入問題,可以採取在執行方法時候,獲得bean來避免錯誤發生。
  20.         CronTriggerBean trigger = (CronTriggerBean) scheduler.getTrigger("cronTrigger", Scheduler.DEFAULT_GROUP);  
  21.         logger.info("*********** trigger: " + trigger);  
  22.         String dbCronExpression = getCronExpressionFromDB();  
  23.         logger.info("*********** dbCronExpression: " + dbCronExpression);  
  24.         String originConExpression = trigger.getCronExpression();  
  25.         logger.info("*********** originConExpression: " + originConExpression);  
  26.         // 判斷從DB中取得的任務時間(dbCronExpression)和現在的quartz執行緒中的任務時間(originConExpression)是否相等
  27.         // 如果相等,則表示使用者並沒有重新設定資料庫中的任務時間,這種情況不需要重新rescheduleJob
  28.         if(!originConExpression.equalsIgnoreCase(dbCronExpression)){  
  29.             try{  
  30.                 trigger.setCronExpression(dbCronExpression);  
  31.                 scheduler.rescheduleJob("cronTrigger", Scheduler.DEFAULT_GROUP, trigger);  
  32.             } catch (ParseException e) {  
  33.                 logger.error("------------------- ParseException Error! -------------------");  
  34.                 e.printStackTrace();  
  35.                 logger.error("-------------------------------------------------------------");  
  36.             }  
  37.         }  
  38.         //執行task
  39.         logger.info("task start time: " + new Date());  
  40.         System.out.println("Task test success!");  
  41.         logger.info("  task end time: " + new Date());  
  42.     }  
  43.     private String getCronExpressionFromDB(){  
  44.         String sql = "select CRON from t_test_task_trigger where available = 1 and trigger_name = 'cronTrigger'";  
  45.         return emsService.getCron(sql);  
  46.     }  
  47. }  

3、Quartz在Spring中動態設定cronTrigger方法二
在2中我們已經能夠實現動態配置cronException,但是我們依然需要設定一個預設的cronException:
          Java程式碼  收藏程式碼
  1. <property name="cronExpression">  
  2.               <value>0/10 * * * * ?</value>  
  3.           </property>  

如果我們拿掉它,則容器(如Jboss)會報錯。
實際上我們希望容器啟動時就去資料庫獲得dbCronException,而不需要再初始化一個cronException。觀察CronTriggerBean,需要初始化cronException,我們可以建立類InitCronTrigger繼承CronTriggerBean,從DB中獲得資料初始化cronException,這樣問題就解決了。
(1)、Spring配置檔案quartz.xml:
Java程式碼  收藏程式碼
  1. <?xml version="1.0" encoding="UTF-8"?>  
  2.         <!DOCTYPE beans PUBLIC "-//SPRING//DTD BEAN//EN""http://www.springframework.org/dtd/spring-beans.dtd">  
  3. <beans>  
  4.     <bean id="scheduleInfoAction"class="wym.task.ScheduleInfoAction">  
  5.         <property name="scheduler" ref="schedulerFactory"/>  
  6.         <property name="emsService" ref="EmsService"/>  
  7.      </bean>  
  8.      <bean id="jobDetail"class="org.springframework.scheduling.quartz.MethodInvokingJobDetailFactoryBean">  
  9.          <property name="targetObject" ref="scheduleInfoAction"/>  
  10.          <property name="targetMethod" value="reScheduleJob"/>  
  11.          <!-- concurrent設為false,多個job不會併發執行 -->  
  12.          <property name="concurrent" value="false"/>  
  13.      </bean>  
  14.      <bean id="cronTrigger"class="wym.task.InitCronTrigger">  
  15.           <property name="jobDetail" ref="jobDetail"/>  
  16.          <!--<property name="cronExpression">-->  
  17.              <!--<value>0/30 * * * * ?</value>-->  
  18.          <!--</property>-->  
  19.           <property name="emsService" ref="EmsService"/>  
  20.       </bean>  
  21.      <bean id="schedulerFactory"class="org.springframework.scheduling.quartz.SchedulerFactoryBean">  
  22.          <property name="triggers">  
  23.              <list>  
  24.                  <ref local="cronTrigger"/>  
  25.              </list>  
  26.          </property>  
  27.     </bean>  
  28. </beans>  

(2)、類InitCronTrigger
注意:在注入scheduleInfoManager屬性的時候,我們可以去讀取DB任務時間(之所以放在setter方法中,是因為需要在設定scheduleInfoManager後進行getCronExpressionFromDB(),否則,也可以①②邏輯把放在類的建構函式中).
注意InitializingCronTrigger必須extends CronTriggerBean.
Java程式碼  收藏程式碼
  1. import com.erry.tntops.ems.service.EmsService;  
  2. import org.springframework.scheduling.quartz.CronTriggerBean;  
  3. import java.io.Serializable;  
  4. import java.text.ParseException;  
  5. publicclass InitCronTrigger extends CronTriggerBean implements Serializable {  
  6.     private EmsService emsService;  
  7.     publicvoid setEmsService(EmsService emsService) throws ParseException {  
  8.         this.emsService = emsService;  
  9.         String cronException = getCronExceptionDB();  
  10.         setCronExpression(cronException);  
  11.     }  
  12.     private String getCronExceptionDB(){  
  13.         String sql = "select CRON from t_test_task_trigger where available = 1 and trigger_name = 'cronTrigger'";  
  14.         System.out.println("*****" + sql);  
  15.         return emsService.getCron(sql);  
  16.     }