最流行的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=""
然后xsi:schemalocation多加下面的内容、
最后是我们的task任务扫描注解
<task:annotation-driven/>
我的配置扫描位置是:
<bean class="org.springframework.beans.factory.annotation.autowiredannotationbeanpostprocessor"/>
<context:component-scan base-package="com.test"/>
扫描的是com.test这样的包下的内容、
下面需要接口和实现(我的这几个java文件都是com.test的包下的、)
public interface imytestservice { public void mytest(); }
@component //import org.springframework.stereotype.component; public class mytestserviceimpl implements imytestservice { @scheduled(cron="0/5 * * * * ? ") //每5秒执行一次 @override public void mytest(){ system.out.println("进入测试"); } }
执行后控制台就会打印出 进入测试 了
需要注意的几点:
1、spring的@scheduled注解 需要写在实现上、
2、 定时器的任务方法不能有返回值(如果有返回值,spring初始化的时候会告诉你有个错误、需要设定一个proxytargetclass的某个值为true、具体就去百度google吧)
3、实现类上要有组件的注解@component
剩下的就是corn表达式了、具体使用以及参数请百度google、
【秒】 【分】 【时】 【日】 【月】 【周】 【年】
下面只例出几个式子
cron表达式 含义
"0 0 12 * * ?" 每天中午十二点触发
"0 15 10 ? * *" 每天早上10:15触发
"0 15 10 * * ?" 每天早上10:15触发
"0 15 10 * * ? *" 每天早上10:15触发
"0 15 10 * * ? 2005" 2005年的每天早上10:15触发
"0 * 14 * * ?" 每天从下午2点开始到2点59分每分钟一次触发
"0 0/5 14 * * ?" 每天从下午2点开始到2:55分结束每5分钟一次触发
"0 0/5 14,18 * * ?" 每天的下午2点至2:55和6点至6点55分两个时间段内每5分钟一次触发
"0 0-5 14 * * ?" 每天14:00至14:05每分钟一次触发
"0 10,44 14 ? 3 wed" 三月的每周三的14:10和14:44触发
"0 15 10 ? * mon-fri" 每个周一、周二、周三、周四、周五的10:15触发
有时候我们的任务(job)需要再某些任务完成之后才能进行;例如从旧的数据库批量导数据的时候;需要现将被其他数据依赖的数据导入新的数据库;然后再进行关系的导入.。在这种情况下我们就可以使用quartz的listener来做文章了。
首先我们写一个主任务的类,命名为mainjob;她的作用是作为一系列任务的开始点。
mainjob.java
package jobs; import org.apache.log4j.logger; import org.quartz.jobexecutioncontext; import org.quartz.jobexecutionexception; import org.springframework.scheduling.quartz.quartzjobbean; public class mainjob extends quartzjobbean { private logger logger=logger.getlogger(getclass()); @override protected void executeinternal(jobexecutioncontext arg0) throws jobexecutionexception { // todo auto-generated method stub logger.debug("just say hi."); } }
然后我们新建另外一个任务(secondjob)作为后续任务:
secondjob.java
package jobs; import org.apache.log4j.logger; import org.quartz.jobexecutioncontext; import org.quartz.jobexecutionexception; import org.springframework.scheduling.quartz.quartzjobbean; public class secondjob extends quartzjobbean { private logger logger=logger.getlogger(getclass()); @override protected void executeinternal(jobexecutioncontext arg0) throws jobexecutionexception { // todo auto-generated method stub logger.debug("i'm the second job."); } }
创建一个triggerlistener,重写其triggercomplete方法,并且添加一些方便spring注入的属性和方法。
nextjobtriggerlistener.java
package listeners; import org.apache.log4j.logger; import org.quartz.jobdetail; import org.quartz.jobexecutioncontext; import org.quartz.scheduler; import org.quartz.trigger; import org.quartz.listeners.triggerlistenersupport; import org.springframework.scheduling.quartz.quartzjobbean; import org.springframework.scheduling.quartz.simpletriggerbean; public class nextjobtriggerlistener extends triggerlistenersupport { private logger logger=logger.getlogger(getclass()); private string name; public string getname() { return this.name; } public void setname(string name) { this.name=name; } private simpletriggerbean nexttrigger; public void setnexttrigger(simpletriggerbean nexttrigger) { this.nexttrigger=nexttrigger; } @override public void triggercomplete(trigger trigger, jobexecutioncontext context, int code) { try{ scheduler schduler=context.getscheduler(); jobdetail nextjob=nexttrigger.getjobdetail(); //查找名称和即将加入的任务一样的任务 jobdetail oldjob=schduler.getjobdetail(nextjob.getname(),nextjob.getgroup()); //查找名称和即将加入的触发器一样的触发器 trigger oldtrigger=schduler.gettrigger(nexttrigger.getname(),nexttrigger.getgroup()); if(oldjob==null&&oldtrigger==null)//同名的任务和触发器都不存在 { logger.debug("inside schedulejob."+code); schduler.schedulejob(nextjob,nexttrigger); }else//同名的任务或触发器 { logger.debug("oldjob==null:"+(oldjob==null)); logger.debug("oldtrigger==null:"+(oldtrigger==null)); } super.triggercomplete(trigger, context, code); }catch(exception e) { e.printstacktrace(); } } }
配置spring 的applicationcontext.xml
applicationcontext.xml
<?xml version="1.0"encoding="utf-8"?> <beans xmlns="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/jee http://www.springframework.org/schema/jee/spring-jee-2.0.xsd"> <!-- 主任务 --> <bean id="mainjob" class="org.springframework.scheduling.quartz.jobdetailbean"> <!-- 运行的类 --> <property name="jobclass"> <value> jobs.mainjob </value> </property> </bean> <!-- 主任务的监听器 --> <bean id="maintriggerlistener" class="listeners.nextjobtriggerlistener"> <!-- 下个触发器 --> <property name="nexttrigger" ref="secondtrigger"></property> <!-- 监听器名称 --> <property name="name" value="maintriggerlistener"></property> </bean> <!-- 主任务的触发器 --> <bean id="maintrigger" class="org.springframework.scheduling.quartz.simpletriggerbean"> <property name="jobdetail"> <!-- 上面创建的任务调度对象 --> <ref bean="mainjob" /> </property> <!-- 启动60秒后执行任务调度的excute方法 --> <property name="startdelay"> <value> 6000 </value> </property> <!-- 运行次数 --> <property name="repeatcount"> <value> </value> </property> <!-- 隔一个小时运行一次(貌似多余,不写会报错) --> <property name="repeatinterval"> <value> 3600000 </value> </property> <property name="triggerlistenernames"> <list> <value> maintriggerlistener </value> </list> </property> </bean> <!-- 后续任务 --> <bean id="secondjob" class="org.springframework.scheduling.quartz.jobdetailbean"> <!-- 运行的类 --> <property name="jobclass"> <value> jobs.secondjob </value> </property> </bean> <!-- 后续任务的触发器 --> <bean id="secondtrigger" class="org.springframework.scheduling.quartz.simpletriggerbean"> <property name="jobdetail"> <!-- 上面创建的任务调度对象 --> <ref bean="secondjob" /> </property> <!-- 启动6秒后执行任务调度的excute方法 --> <property name="startdelay"> <value> 6000 </value> </property> <!-- 运行次数 --> <property name="repeatcount"> <value> </value> </property> <!-- 隔一个小时运行一次(貌似多余,不写会报错) --> <property name="repeatinterval"> <!-- <value>3600000</value> --> <value> 6000 </value> </property> </bean> <!-- 任务调度工厂类 --> <bean class="org.springframework.scheduling.quartz.schedulerfactorybean"> <!-- 这一部分的配置不用管 --> <property name="quartzproperties"> <props> <prop key="org.quartz.threadpool.class"> org.quartz.simpl.simplethreadpool </prop> <prop key="org.quartz.threadpool.threadcount"> </prop> <prop key="org.quartz.threadpool.threadpriority"> </prop> <prop key="org.quartz.threadpool.threadsinheritcontextclassloaderofinitializingthread"> true </prop> </props> </property> <!-- 触发器,可以放一大堆触发器 --> <property name="triggers"> <list> <!-- 在这里加 --> <ref bean="maintrigger"/> </list> </property> <property name="triggerlisteners"> <list> <!-- 触发器的监听器 --> <ref bean="maintriggerlistener" /> </list> </property> </bean> </beans>
开启服务器,输出
debug [ mainjob.executeinternal(14) ] just say hi. debug [ nextjobtriggerlistener.triggercomplete(38) ] inside schedulejob .3 debug [secondjob.executeinternal(14)] i'm the second job. debug [ nextjobtriggerlistener.triggercomplete(43) ] oldjob==null:false debug [ nextjobtriggerlistener.triggercomplete(44) ] oldtrigger== null:false
另外这里一个任务只绑定了一个简单的触发器,这样做是为了比较方便地可以检测到任务完成的情况;至于任务的具体内容就任由大家发挥了。写这篇文章希望能有人在其中获得启发。