最流行的java后台框架spring quartz定时任务
配置quartz在spring中需要三个jar包:
quartz-1.8.5.jar、commons-collections-3.2.1.jar、commons-logging-1.1.jar
首先要配置我们的spring.xml
xmlns多加下面的内容、
xmlns:task="http://www.springframework.org/schema/task"
然后xsi:schemaLocation多加下面的内容、
http://www.springframework.org/schema/task
http://www.springframework.org/schema/task/spring-task-3.1.xsd
最后是我们的task任务扫描注解
<task:annotation-driven/>
我的配置扫描位置是:
<context:annotation-config/> <beanclass="org.springframework.beans.factory.annotation.AutowiredAnnotationBeanPostProcessor"/> <context:component-scanbase-package="com.test"/>
扫描的是com.test这样的包下的内容、
下面需要接口和实现(我的这几个java文件都是com.test的包下的、)
publicinterfaceIMyTestService{ publicvoidmyTest(); }
@Component//importorg.springframework.stereotype.Component; publicclassMyTestServiceImplimplementsIMyTestService{ @Scheduled(cron="0/5****?")//每5秒执行一次 @Override publicvoidmyTest(){ System.out.println("进入测试"); } }
执行后控制台就会打印出 进入测试 了
需要注意的几点:
1、spring的@Scheduled注解 需要写在实现上、
2、定时器的任务方法不能有返回值(如果有返回值,spring初始化的时候会告诉你有个错误、需要设定一个proxytargetclass的某个值为true、具体就去百度google吧)
3、实现类上要有组件的注解@Component
剩下的就是corn表达式了、具体使用以及参数请百度google、
【秒】 【分】 【时】 【日】 【月】 【周】 【年】
下面只例出几个式子
CRON表达式 含义
"0012**?" 每天中午十二点触发
"01510?**" 每天早上10:15触发
"01510**?" 每天早上10:15触发
"01510**?*" 每天早上10:15触发
"01510**?2005" 2005年的每天早上10:15触发
"0*14**?" 每天从下午2点开始到2点59分每分钟一次触发
"00/514**?" 每天从下午2点开始到2:55分结束每5分钟一次触发
"00/514,18**?" 每天的下午2点至2:55和6点至6点55分两个时间段内每5分钟一次触发
"00-514**?" 每天14:00至14:05每分钟一次触发
"010,4414?3WED" 三月的每周三的14:10和14:44触发
"01510?*MON-FRI" 每个周一、周二、周三、周四、周五的10:15触发
有时候我们的任务(Job)需要再某些任务完成之后才能进行;例如从旧的数据库批量导数据的时候;需要现将被其他数据依赖的数据导入新的数据库;然后再进行关系的导入.。在这种情况下我们就可以使用Quartz的listener来做文章了。
首先我们写一个主任务的类,命名为MainJob;她的作用是作为一系列任务的开始点。
MainJob.java
packagejobs; importorg.apache.log4j.Logger; importorg.quartz.JobExecutionContext; importorg.quartz.JobExecutionException; importorg.springframework.scheduling.quartz.QuartzJobBean; publicclassMainJobextendsQuartzJobBean{ privateLoggerlogger=Logger.getLogger(getClass()); @Override protectedvoidexecuteInternal(JobExecutionContextarg0) throwsJobExecutionException{ //TODOAuto-generatedmethodstub logger.debug("Justsayhi."); } }
然后我们新建另外一个任务(SecondJob)作为后续任务:
SecondJob.java
packagejobs; importorg.apache.log4j.Logger; importorg.quartz.JobExecutionContext; importorg.quartz.JobExecutionException; importorg.springframework.scheduling.quartz.QuartzJobBean; publicclassSecondJobextendsQuartzJobBean{ privateLoggerlogger=Logger.getLogger(getClass()); @Override protectedvoidexecuteInternal(JobExecutionContextarg0) throwsJobExecutionException{ //TODOAuto-generatedmethodstub logger.debug("I'mthesecondjob."); } }
创建一个TriggerListener,重写其triggerComplete方法,并且添加一些方便spring注入的属性和方法。
NextJobTriggerListener.java
packagelisteners; importorg.apache.log4j.Logger; importorg.quartz.JobDetail; importorg.quartz.JobExecutionContext; importorg.quartz.Scheduler; importorg.quartz.Trigger; importorg.quartz.listeners.TriggerListenerSupport; importorg.springframework.scheduling.quartz.QuartzJobBean; importorg.springframework.scheduling.quartz.SimpleTriggerBean; publicclassNextJobTriggerListenerextendsTriggerListenerSupport{ privateLoggerlogger=Logger.getLogger(getClass()); privateStringname; publicStringgetName(){ returnthis.name; } publicvoidsetName(Stringname) { this.name=name; } privateSimpleTriggerBeannextTrigger; publicvoidsetNextTrigger(SimpleTriggerBeannextTrigger){ this.nextTrigger=nextTrigger; } @Override publicvoidtriggerComplete(Triggertrigger,JobExecutionContextcontext,intcode){ try{ Schedulerschduler=context.getScheduler(); JobDetailnextJob=nextTrigger.getJobDetail(); //查找名称和即将加入的任务一样的任务 JobDetailoldJob=schduler.getJobDetail(nextJob.getName(),nextJob.getGroup()); //查找名称和即将加入的触发器一样的触发器 TriggeroldTrigger=schduler.getTrigger(nextTrigger.getName(),nextTrigger.getGroup()); if(oldJob==null&&oldTrigger==null)//同名的任务和触发器都不存在 { logger.debug("insidescheduleJob."+code); schduler.scheduleJob(nextJob,nextTrigger); }else//同名的任务或触发器 { logger.debug("oldJob==null:"+(oldJob==null)); logger.debug("oldTrigger==null:"+(oldTrigger==null)); } super.triggerComplete(trigger,context,code); }catch(Exceptione) { e.printStackTrace(); } } }
配置spring的applicationContext.xml
applicationContext.xml
<?xmlversion="1.0"encoding="UTF-8"?> <beansxmlns="http://www.springframework.org/schema/beans" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:jee="http://www.springframework.org/schema/jee" xsi:schemaLocation="http://www.springframework.org/schema/beanshttp://www.springframework.org/schema/beans/spring-beans-2.0.xsd http://www.springframework.org/schema/jeehttp://www.springframework.org/schema/jee/spring-jee-2.0.xsd"> <!--主任务--> <beanid="mainJob" class="org.springframework.scheduling.quartz.JobDetailBean"> <!--运行的类--> <propertyname="jobClass"> <value>jobs.MainJob</value> </property> </bean> <!--主任务的监听器--> <beanid="mainTriggerListener" class="listeners.NextJobTriggerListener"> <!--下个触发器--> <propertyname="nextTrigger"ref="secondTrigger"></property> <!--监听器名称--> <propertyname="name"value="mainTriggerListener"></property> </bean> <!--主任务的触发器--> <beanid="mainTrigger" class="org.springframework.scheduling.quartz.SimpleTriggerBean"> <propertyname="jobDetail"> <!--上面创建的任务调度对象--> <refbean="mainJob"/> </property> <!--启动60秒后执行任务调度的excute方法--> <propertyname="startDelay"> <value>6000</value> </property> <!--运行次数--> <propertyname="repeatCount"> <value></value> </property> <!--隔一个小时运行一次(貌似多余,不写会报错)--> <propertyname="repeatInterval"> <value>3600000</value> </property> <propertyname="triggerListenerNames"> <list> <value>mainTriggerListener</value> </list> </property> </bean> <!--后续任务--> <beanid="secondJob" class="org.springframework.scheduling.quartz.JobDetailBean"> <!--运行的类--> <propertyname="jobClass"> <value>jobs.SecondJob</value> </property> </bean> <!--后续任务的触发器--> <beanid="secondTrigger" class="org.springframework.scheduling.quartz.SimpleTriggerBean"> <propertyname="jobDetail"> <!--上面创建的任务调度对象--> <refbean="secondJob"/> </property> <!--启动6秒后执行任务调度的excute方法--> <propertyname="startDelay"> <value>6000</value> </property> <!--运行次数--> <propertyname="repeatCount"> <value></value> </property> <!--隔一个小时运行一次(貌似多余,不写会报错)--> <propertyname="repeatInterval"> <!-- <value>3600000</value> --> <value>6000</value> </property> </bean> <!--任务调度工厂类--> <bean class="org.springframework.scheduling.quartz.SchedulerFactoryBean"> <!--这一部分的配置不用管--> <propertyname="quartzProperties"> <props> <propkey="org.quartz.threadPool.class"> org.quartz.simpl.SimpleThreadPool </prop> <propkey="org.quartz.threadPool.threadCount"></prop> <propkey="org.quartz.threadPool.threadPriority"> </prop> <prop key="org.quartz.threadPool.threadsInheritContextClassLoaderOfInitializingThread"> true </prop> </props> </property> <!--触发器,可以放一大堆触发器--> <propertyname="triggers"> <list> <!--在这里加--> <refbean="mainTrigger"/> </list> </property> <propertyname="triggerListeners"> <list> <!--触发器的监听器--> <refbean="mainTriggerListener"/> </list> </property> </bean> </beans>
开启服务器,输出
DEBUG[MainJob.executeInternal(14)]Justsayhi. DEBUG[NextJobTriggerListener.triggerComplete(38)]insidescheduleJob.3 DEBUG[SecondJob.executeInternal(14)]I'mthesecondjob. DEBUG[NextJobTriggerListener.triggerComplete(43)]oldJob==null:false DEBUG[NextJobTriggerListener.triggerComplete(44)]oldTrigger==null:false
另外这里一个任务只绑定了一个简单的触发器,这样做是为了比较方便地可以检测到任务完成的情况;至于任务的具体内容就任由大家发挥了。写这篇文章希望能有人在其中获得启发。