欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页  >  IT编程

最流行的java后台框架spring quartz定时任务

程序员文章站 2024-03-08 11:12:16
配置quartz 在spring中需要三个jar包: quartz-1.8.5.jar、commons-collections-3.2.1.jar、commons-log...

配置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/> 

我的配置扫描位置是:

复制代码 代码如下:
<context:annotation-config/> 
    <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

另外这里一个任务只绑定了一个简单的触发器,这样做是为了比较方便地可以检测到任务完成的情况;至于任务的具体内容就任由大家发挥了。写这篇文章希望能有人在其中获得启发。