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

 更新时间:2015年12月01日 08:50:26   投稿:hebedich  
近日项目开发中需要执行一些定时任务,比如需要在每天凌晨时候,分析一次前一天的日志信息,借此机会整理了一下定时任务的几种实现方式,由于项目采用spring框架,所以我都将结合spring框架来介绍

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

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

相关文章

  • Java 数据交换 Json 和 异步请求 Ajax详解

    Java 数据交换 Json 和 异步请求 Ajax详解

    Json(JavaScript Object Notation)是一种轻量级的数据交换格式,采用键值对的形式来表示数据,它广泛应用于Web开发中,特别适合于前后端数据传输和存储,这篇文章主要介绍了Java数据交换Json和异步请求Ajax,需要的朋友可以参考下
    2023-09-09
  • Java的类型擦除式泛型详解

    Java的类型擦除式泛型详解

    Java语言中的泛型只存在于程序源码之中,在编译后的字节码文件里,则全部泛型都会被替换为原来的原始类型(Raw Type),并且会在相应的地方插入强制转型的代码。这篇文章主要介绍了Java的类型擦除式泛型,需要的朋友可以参考下
    2021-08-08
  • 关于BufferedReader读取文件指定字符集问题

    关于BufferedReader读取文件指定字符集问题

    这篇文章主要介绍了关于BufferedReader读取文件指定字符集问题,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2021-12-12
  • Java字符串比较方法equals的空指针异常的解决

    Java字符串比较方法equals的空指针异常的解决

    这篇文章主要介绍了Java字符串比较方法equals的空指针异常的解决,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2019-09-09
  • @GrpcServise 注解的作用和使用示例详解

    @GrpcServise 注解的作用和使用示例详解

    @GrpcService 是一个 Spring Boot 处理器,它会查找实现了 grpc::BindableService 接口的类,并将其包装成一个 Spring Bean 对象,这篇文章主要介绍了@GrpcServise 注解的作用和使用,需要的朋友可以参考下
    2023-05-05
  • java中BigDecimal用法详解

    java中BigDecimal用法详解

    本文详细讲解了java中BigDecimal的用法,文中通过示例代码介绍的非常详细。对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
    2021-12-12
  • jsp中EL表达式获取数据

    jsp中EL表达式获取数据

    EL 全名为Expression Language。本文给大家介绍的是在jsp中EL表达式获取数据的几种方式,希望大家能够喜欢
    2016-07-07
  • SpringBoot访问MongoDB数据库的两种方式

    SpringBoot访问MongoDB数据库的两种方式

    MongoDB是一种非关系型数据库,通过文档存储数据,适用于大规模数据存储和高并发访问,这篇文章主要介绍了SpringBoot访问MongoDB数据库的两种方式,感兴趣想要详细了解可以参考下文
    2023-05-05
  • 详解Spring循环依赖的解决方案

    详解Spring循环依赖的解决方案

    这篇文章主要介绍了详解Spring循环依赖的解决方案,小编觉得挺不错的,现在分享给大家,也给大家做个参考。一起跟随小编过来看看吧
    2018-05-05
  • Java中使用BeanMap将对象转为Map详解

    Java中使用BeanMap将对象转为Map详解

    这篇文章主要介绍了Java中使用BeanMap将对象转为Map详解,BeanMap 是 Apache Commons BeanUtils 库中的一个类,BeanMap 可以将 Java 对象的属性作为键,属性值作为对应的值,存储在一个 Map 中,它提供了一种将 Java 对象转换为 Map 的方式,需要的朋友可以参考下
    2024-01-01

最新评论