Spring Quartz定时器 的动态调度

秦怀卓 2012-10-28

众所周知spring的Quartz定时器的功能非常强大,可以在特定的年月日小时分秒的时间点触发时间,完成事件的调度,就像windows中得计划任务一样。下面看一个典型的Quartz定时器的实现:

1、首先实现一个用于被执行的类,这个类用于被定时器调度,这个类不需要继承任何类或者接口,代码如下:

publicTestQuartz{

publicvoiddoSomething(){

//TODO

}

}

2、spring配置文件,具体配置

<!--被执行类-->

<beanid="testQuarzt"class="test.testQuarzt"/>

<!--将testQuarzt注入到job中-->

<beanid="testQuartzJob"class="org.springframework.scheduling.quartz.MethodInvokingJobDetailFactoryBean">

<propertyname="targetObject"ref="testQuarzt"/>

<propertyname="targetMethod"value="doSomething"/>

<propertyname="concurrent"value="false"/>

</bean>

<!--将job注入到定时触发器-->

<beanid="testTrigger"class="org.springframework.scheduling.quartz.CronTriggerBean">

<propertyname="jobDetail"ref="testQuartzJob"/>

<propertyname="cronExpression">

<value>006**?<value>

<property>

<bean>

<!--将触发器注入任务工程-->

<beanclass="org.springframework.scheduling.quartz.SchedulerFactoryBean">

<propertyname="triggers">

<list>

<refbean="testTrigger"/>

</list>

</property>

</bean>

这样一来,一个简单的SpringQuartz定时器就完成了,这个定时器会在每天早上6点去调testQuarzt类中得doSomething方法,至于为什么是早上六点钟将在最后解释。

但是假设这个定时器部署在tomcat中,如果我们要修改定时器的调度时间,就必须修改spring配置文件,再重启tomcat,事实上,有很多时候我们没有条件去重启tomcat,或者,有特定的需求,需要前台可以直接修改调度时间,这样一来,问题就出现了。

但是我们细心观察,不难发现,调度的时间其实是org.springframework.scheduling.quartz.CronTriggerBean这个类里面的cronExpression属性控制着定时器的调度时间,而事实上,spring肯定在加载配置文件的时候,生成org.springframework.scheduling.quartz.SchedulerFactoryBean的对象,org.springframework.scheduling.quartz.CronTriggerBean的对象,如果我们可以获取这两个对象,问题似乎可以解决了,事实上,我们获取这个对象是非常容易的,只要将这两个对象注入到我们的类中,而且,事实上,在SchedulerFactoryBean中提供了很多方法可以方便我们进行操作。

接下来将作如下修改:

1、修改TestQuartz类,代码如下:

publicTestQuartz{

privateSchedulerscheduler;

//设值注入,通过setter方法传入被调用者的实例scheduler

publicvoidsetScheduler(Schedulerscheduler){

this.scheduler=scheduler;

}

publicvoiddoSomething(){

//TODO

}

publicvoidresetJob(StringcronExpression){

//运行时可通过动态注入的scheduler得到trigger,注意采用这种注入方式在有的项目中会有问题,如果遇到注入问题,可以采取在运行方法时候,获得bean来避免错误发生。

CronTriggerBeantrigger=(CronTriggerBean)scheduler.getTrigger("testTrigger",Scheduler.DEFAULT_GROUP);

StringoriginConExpression=trigger.getCronExpression();

//如果相等,则表示用户并没有重新设定数据库中的任务时间,这种情况不需要重新rescheduleJob

if(!originConExpression.equalsIgnoreCase(cronExpression)){

trigger.setCronExpression(cronExpression);

scheduler.rescheduleJob("testTrigger",Scheduler.DEFAULT_GROUP,trigger);

}

}

}

2、spring配置文件的修改,如下:

<!--被执行类-->

<beanid="testQuarzt"class="test.testQuarzt">

<propertyname="scheduler"ref="schedulerFactory"/>

</bean>

<!--将testQuarzt注入到job中-->

<beanid="testQuartzJob"class="org.springframework.scheduling.quartz.MethodInvokingJobDetailFactoryBean">

<propertyname="targetObject"ref="testQuarzt"/>

<propertyname="targetMethod"value="doSomething"/>

<propertyname="concurrent"value="false"/>

</bean>

<!--将job注入到定时触发器-->

<beanid="testTrigger"class="org.springframework.scheduling.quartz.CronTriggerBean">

<propertyname="jobDetail"ref="testQuartzJob"/>

<propertyname="cronExpression">

<value>006**?<value>

<property>

<bean>

<!--将触发器注入任务工程-->

<beanid="schedulerFactory"class="org.springframework.scheduling.quartz.SchedulerFactoryBean">

<propertyname="triggers">

<list>

<refbean="testTrigger"/>

</list>

</property>

</bean>

如此一来,这个可以动态调度的定时器就完成了,接下来,将以下那个调度时间格式的问题。

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

秒0-59,-*/

分0-59,-*/

小时0-23,-*/

日期1-31,-*?/LWC

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

星期1-7或者SUN-SAT,-*?/LC#

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

0012**?---------------在每天中午12:00触发

01510?**---------------每天上午10:15触发

01510**?---------------每天上午10:15触发

01510**?*---------------每天上午10:15触发

01510**?2005---------------在2005年中的每天上午10:15触发

0*14**?---------------每天在下午2:00至2:59之间每分钟触发一次

00/514**?---------------每天在下午2:00至2:59之间每5分钟触发一次

00/514,18**?---------------每天在下午2:00至2:59和6:00至6:59之间的每5分钟触发一次

00-514**?---------------每天在下午2:00至2:05之间每分钟触发一次

010,4414?3WED---------------每三月份的星期三在下午2:00和2:44时触发

01510?*MON-FRI---------------从星期一至星期五的每天上午10:15触发

0151015*?---------------在每个月的每15天的上午10:15触发

01510L*?---------------在每个月的最后一天的上午10:15触发

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

01510?*6L2002-2005---------------在2002,2003,2004and2005年的每个月的最后一个星期五的上午10:15触发

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

00121/5*?---------------从每月的第一天起每过5天的中午12:00时触发

011111111?---------------在每个11月11日的上午11:11时触发.?

参考资料:

1.http://www.springframework.orgSpring的官方网站

2.http://www.opensymphony.com/quartz/Quartz的官方网站

相关推荐