Spring Quartz定时器 的动态调度
众所周知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的官方网站