`
langgufu
  • 浏览: 2309232 次
  • 性别: Icon_minigender_1
  • 来自: 北京
社区版块
存档分类
最新评论

Spring,jdk定时任务的几种实现以及任务线程是串行还是并行执行(转载)

阅读更多

近日项目开发中需要执行一些定时任务,比如需要在每天凌晨时候,分析一次前一天的日志信息,借此机会整理了一下定时任务的几种实现方式,由于项目采用spring框架,所以我都将结合

spring框架来介绍。

一.分类

  • 从实现的技术上来分类,目前主要有三种技术(或者说有三种产品):

  1. Java自带的java.util.Timer类,这个类允许你调度一个java.util.TimerTask任务。使用这种方式可以让你的程序按照某一个频度执行,但不能在指定时间运行。一般用的较少,这篇文章将不做详细介绍。
  2. 使用Quartz,这是一个功能比较强大的的调度器,可以让你的程序在指定时间执行,也可以按照某一个频度执行,配置起来稍显复杂,稍后会详细介绍。
  3. Spring3.0以后自带的task,可以将它看成一个轻量级的Quartz,而且使用起来比Quartz简单许多,稍后会介绍。
  • 从作业类的继承方式来讲,可以分为两类:

  1. 作业类需要继承自特定的作业类基类,如Quartz中需要继承自org.springframework.scheduling.quartz.QuartzJobBean;java.util.Timer中需要继承自java.util.TimerTask。
  2. 作业类即普通的java类,不需要继承自任何基类。

注:个人推荐使用第二种方式,因为这样所以的类都是普通类,不需要事先区别对待。

 

  • 从任务调度的触发时机来分,这里主要是针对作业使用的触发器,主要有以下两种:
  1. 每隔指定时间则触发一次,在Quartz中对应的触发器为:org.springframework.scheduling.quartz.SimpleTriggerBean
  2. 每到指定时间则触发一次,在Quartz中对应的调度器为:org.springframework.scheduling.quartz.CronTriggerBean

注:并非每种任务都可以使用这两种触发器,如java.util.TimerTask任务就只能使用第一种。Quartz和spring task都可以支持这两种触发条件。

 

 

二.用法说明

详细介绍每种任务调度工具的使用方式,包括Quartz和spring task两种。

Quartz

第一种,作业类继承自特定的基类:org.springframework.scheduling.quartz.QuartzJobBean。

第一步:定义作业类

 

Java代码   收藏代码
  1. import org.quartz.JobExecutionContext;  
  2. import org.quartz.JobExecutionException;  
  3. import org.springframework.scheduling.quartz.QuartzJobBean;  
  4. public class Job1 extends QuartzJobBean {  
  5.   
  6. private int timeout;  
  7. private static int i = 0;  
  8. //调度工厂实例化后,经过timeout时间开始执行调度  
  9. public void setTimeout(int timeout) {  
  10. this.timeout = timeout;  
  11. }  
  12.   
  13. /** 
  14. * 要调度的具体任务 
  15. */  
  16. @Override  
  17. protected void executeInternal(JobExecutionContext context)  
  18. throws JobExecutionException {  
  19.   System.out.println("定时任务执行中…");  
  20. }  
  21. }  

 第二步:spring配置文件中配置作业类JobDetailBean

Xml代码   收藏代码
  1. <bean name="job1" class="org.springframework.scheduling.quartz.JobDetailBean">  
  2. <property name="jobClass" value="com.gy.Job1" />  
  3. <property name="jobDataAsMap">  
  4. <map>  
  5. <entry key="timeout" value="0" />  
  6. </map>  
  7. </property>  
  8. </bean>  

 说明:org.springframework.scheduling.quartz.JobDetailBean有两个属性,jobClass属性即我们在java代码中定义的任务类,jobDataAsMap属性即该任务类中需要注入的属性值。

第三步:配置作业调度的触发方式(触发器)

Quartz的作业触发器有两种,分别是

org.springframework.scheduling.quartz.SimpleTriggerBean

org.springframework.scheduling.quartz.CronTriggerBean

第一种SimpleTriggerBean,只支持按照一定频度调用任务,如每隔30分钟运行一次。

配置方式如下:

 

Xml代码   收藏代码
  1. <bean id="simpleTrigger" class="org.springframework.scheduling.quartz.SimpleTriggerBean">  
  2. <property name="jobDetail" ref="job1" />  
  3. <property name="startDelay" value="0" /><!-- 调度工厂实例化后,经过0秒开始执行调度 -->  
  4. <property name="repeatInterval" value="2000" /><!-- 每2秒调度一次 -->  
  5. </bean>  

第二种CronTriggerBean,支持到指定时间运行一次,如每天12:00运行一次等。

配置方式如下:

Xml代码   收藏代码
  1. <bean id="cronTrigger" class="org.springframework.scheduling.quartz.CronTriggerBean">  
  2. <property name="jobDetail" ref="job1" />  
  3. <!—每天12:00运行一次 -->  
  4. <property name="cronExpression" value="0 0 12 * * ?" />  
  5. </bean>  

 关于cronExpression表达式的语法参见附录。

第四步:配置调度工厂 

Xml代码   收藏代码
  1. <bean class="org.springframework.scheduling.quartz.SchedulerFactoryBean">  
  2. <property name="triggers">  
  3. <list>  
  4. <ref bean="cronTrigger" />  
  5. </list>  
  6. </property>  
  7. </bean>  

 说明:该参数指定的就是之前配置的触发器的名字。

第五步:启动你的应用即可,即将工程部署至tomcat或其他容器。

 

 

第二种,作业类不继承特定基类。

Spring能够支持这种方式,归功于两个类:

org.springframework.scheduling.timer.MethodInvokingTimerTaskFactoryBean

org.springframework.scheduling.quartz.MethodInvokingJobDetailFactoryBean

这两个类分别对应spring支持的两种实现任务调度的方式,即前文提到到java自带的timer task方式和Quartz方式。这里我只写MethodInvokingJobDetailFactoryBean的用法,使用该类的好处是,我们的任务类不再需要继承自任何类,而是普通的pojo。

第一步:编写任务类

Java代码   收藏代码
  1. public class Job2 {  
  2. public void doJob2() {  
  3. System.out.println("不继承QuartzJobBean方式-调度进行中...");  
  4. }  
  5. }  

 可以看出,这就是一个普通的类,并且有一个方法。

第二步:配置作业类

Xml代码   收藏代码
  1. <bean id="job2"  
  2. class="org.springframework.scheduling.quartz.MethodInvokingJobDetailFactoryBean">  
  3. <property name="targetObject">  
  4. <bean class="com.gy.Job2" />  
  5. </property>  
  6. <property name="targetMethod" value="doJob2" />  
  7. <property name="concurrent" value="false" /><!-- 作业不并发调度 -->  
  8. </bean>  

 说明:这一步是关键步骤,声明一个MethodInvokingJobDetailFactoryBean,有两个关键属性:targetObject指定任务类,targetMethod指定运行的方法。往下的步骤就与方法一相同了,为了完整,同样贴出。

第三步:配置作业调度的触发方式(触发器)

Quartz的作业触发器有两种,分别是

org.springframework.scheduling.quartz.SimpleTriggerBean

org.springframework.scheduling.quartz.CronTriggerBean

第一种SimpleTriggerBean,只支持按照一定频度调用任务,如每隔30分钟运行一次。

配置方式如下:

Xml代码   收藏代码
  1. <bean id="simpleTrigger" class="org.springframework.scheduling.quartz.SimpleTriggerBean">  
  2. <property name="jobDetail" ref="job2" />  
  3. <property name="startDelay" value="0" /><!-- 调度工厂实例化后,经过0秒开始执行调度 -->  
  4. <property name="repeatInterval" value="2000" /><!-- 每2秒调度一次 -->  
  5. </bean>  

 第二种CronTriggerBean,支持到指定时间运行一次,如每天12:00运行一次等。

配置方式如下:

Xml代码   收藏代码
  1. <bean id="cronTrigger" class="org.springframework.scheduling.quartz.CronTriggerBean">  
  2. <property name="jobDetail" ref="job2" />  
  3. <!—每天12:00运行一次 -->  
  4. <property name="cronExpression" value="0 0 12 * * ?" />  
  5. </bean>  

以上两种调度方式根据实际情况,任选一种即可。

第四步:配置调度工厂 

Xml代码   收藏代码
  1. <bean class="org.springframework.scheduling.quartz.SchedulerFactoryBean">  
  2. <property name="triggers">  
  3. <list>  
  4. <ref bean="cronTrigger" />  
  5. </list>  
  6. </property>  
  7. </bean>  

说明:该参数指定的就是之前配置的触发器的名字。

第五步:启动你的应用即可,即将工程部署至tomcat或其他容器。

 

到此,spring中Quartz的基本配置就介绍完了,当然了,使用之前,要导入相应的spring的包与Quartz的包,这些就不消多说了。

其实可以看出Quartz的配置看上去还是挺复杂的,没有办法,因为Quartz其实是个重量级的工具,如果我们只是想简单的执行几个简单的定时任务,有没有更简单的工具,有!

请看我第下文Spring task的介绍。

 

Spring-Task

上节介绍了在Spring 中使用Quartz,本文介绍Spring3.0以后自主开发的定时任务工具,spring task,可以将它比作一个轻量级的Quartz,而且使用起来很简单,除spring相关的包外不需要额外的包,而且支持注解和配置文件两种

形式,下面将分别介绍这两种方式。

第一种:配置文件方式

第一步:编写作业类

即普通的pojo,如下:

Java代码   收藏代码
  1. import org.springframework.stereotype.Service;  
  2. @Service  
  3. public class TaskJob {  
  4.       
  5.     public void job1() {  
  6.         System.out.println(“任务进行中。。。”);  
  7.     }  
  8. }  

 第二步:在spring配置文件头中添加命名空间及描述

Xml代码   收藏代码
  1. <beans xmlns="http://www.springframework.org/schema/beans"  
  2.     xmlns:task="http://www.springframework.org/schema/task"   
  3.     。。。。。。  
  4.     xsi:schemaLocation="http://www.springframework.org/schema/task http://www.springframework.org/schema/task/spring-task-3.0.xsd">  

 第三步:spring配置文件中设置具体的任务

Xml代码   收藏代码
  1.  <task:scheduled-tasks>   
  2.         <task:scheduled ref="taskJob" method="job1" cron="0 * * * * ?"/>   
  3. </task:scheduled-tasks>  
  4.   
  5. <context:component-scan base-package=" com.gy.mytask " />  

说明:ref参数指定的即任务类,method指定的即需要运行的方法,cron及cronExpression表达式,具体写法这里不介绍了,详情见上篇文章附录。

<context:component-scan base-package="com.gy.mytask" />这个配置不消多说了,spring扫描注解用的。

到这里配置就完成了,是不是很简单。

第二种:使用注解形式

也许我们不想每写一个任务类还要在xml文件中配置下,我们可以使用注解@Scheduled,我们看看源文件中该注解的定义:

Java代码   收藏代码
  1. @Target({java.lang.annotation.ElementType.METHOD, java.lang.annotation.ElementType.ANNOTATION_TYPE})  
  2. @Retention(RetentionPolicy.RUNTIME)  
  3. @Documented  
  4. public @interface Scheduled  
  5. {  
  6.   public abstract String cron();  
  7.   
  8.   public abstract long fixedDelay();  
  9.   
  10.   public abstract long fixedRate();  
  11. }  

 可以看出该注解有三个方法或者叫参数,分别表示的意思是:

cron:指定cron表达式

fixedDelay:官方文档解释:An interval-based trigger where the interval is measured from the completion time of the previous task. The time unit value is measured in milliseconds.即表示从上一个任务完成开始到下一个任务开始的间隔,单位是毫秒。

fixedRate:官方文档解释:An interval-based trigger where the interval is measured from the start time of the previous task. The time unit value is measured in milliseconds.即从上一个任务开始到下一个任务开始的间隔,单位是毫秒。

 

下面我来配置一下。

第一步:编写pojo

Java代码   收藏代码
  1. import org.springframework.scheduling.annotation.Scheduled;    
  2. import org.springframework.stereotype.Component;  
  3.   
  4. @Component(“taskJob”)  
  5. public class TaskJob {  
  6.     @Scheduled(cron = "0 0 3 * * ?")  
  7.     public void job1() {  
  8.         System.out.println(“任务进行中。。。”);  
  9.     }  
  10. }  

 第二步:添加task相关的配置:

Xml代码   收藏代码
  1. <?xml version="1.0" encoding="UTF-8"?>  
  2. <beans xmlns="http://www.springframework.org/schema/beans"  
  3.     xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:aop="http://www.springframework.org/schema/aop"  
  4.     xmlns:context="http://www.springframework.org/schema/context"  
  5.     xmlns:tx="http://www.springframework.org/schema/tx"  
  6.     xmlns:task="http://www.springframework.org/schema/task"  
  7.     xsi:schemaLocation="  
  8.         http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-3.0.xsd  
  9.         http://www.springframework.org/schema/aop http://www.springframework.org/schema/aop/spring-aop-3.0.xsd  
  10.         http://www.springframework.org/schema/context   
  11. http://www.springframework.org/schema/jdbc/spring-jdbc-3.0.xsd  
  12.         http://www.springframework.org/schema/tx http://www.springframework.org/schema/tx/spring-tx-3.0.xsd  
  13.         http://www.springframework.org/schema/task http://www.springframework.org/schema/task/spring-task-3.0.xsd"  
  14.     default-lazy-init="false">  
  15.   
  16.   
  17.     <context:annotation-config />  
  18.     <!—spring扫描注解的配置   -->  
  19.     <context:component-scan base-package="com.gy.mytask" />  
  20.       
  21. <!—开启这个配置,spring才能识别@Scheduled注解   -->  
  22.     <task:annotation-driven scheduler="qbScheduler" mode="proxy"/>  
  23.     <task:scheduler id="qbScheduler" pool-size="10"/>  

说明:理论上只需要加上<task:annotation-driven />这句配置就可以了,这些参数都不是必须的。

 

 Ok配置完毕,当然spring task还有很多参数,我就不一一解释了,具体参考xsd文档http://www.springframework.org/schema/task/spring-task-3.0.xsd。

附录:

cronExpression的配置说明,具体使用以及参数请百度google

字段   允许值   允许的特殊字符

秒    0-59    , - * /

分    0-59    , - * /

小时    0-23    , - * /

日期    1-31    , - * ? / L W C

月份    1-12 或者 JAN-DEC    , - * /

星期    1-7 或者 SUN-SAT    , - * ? / L C #

年(可选)    留空, 1970-2099    , - * / 

- 区间  

* 通配符  

? 你不想设置那个字段

下面只例出几个式子

 

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触发

 

除了以上三种定时任务方式外,还有线程池定时执行任务方式--Executors.newScheduledThreadPool(1);

 

 

三种定时任务运行方式并行或串行:

1.TimerTask对同一个定时任务是串行执行,对多个暂时不知。

2.spring quartz任务并行,串行

 

 

任务有并行串行之分,并行是指:一个定时任务,当执行时间到了的时候,立刻执行此任务,不管当前这个任务是否在执行中;串行是指:一个定时任务,当执行时间到了的时候,需要等待当前任务执行完毕,再去执行下一个任务。

 

quartz框架中防止任务并行可以有两种方案:

 

1、如果是通过MethodInvokingJobDetailFactoryBean在运行中动态生成的Job,配置的xml文件有个concurrent属性,这个属性的功能是配置此job是否可以并行运行,如果为false则表示不可以并行运行,否则可以并行如果一个job的业务处理发费的时间超过了job的启动的间隔时间(repeatInterval),这个属性非常有用。如果为false,那么,在这种情况下,当前job还在运行,那么下一个job只能延时运行。如果为true,那么job就会并行运行。

 

 

 

     <bean id=" jobCompareB2cAndLocal" class="com.vipshop.management.util.quartz.MethodInvokingJobDetailFactoryBean ">
          <property name="targetObject " ref="delegateJob " />
          <property name="targetMethod " value="方法名" />
          <property name="concurrent " value="false "></property >
    </bean >
 

 

2、如果是通过自定义要执行的任务的类的名称实现job的话,则有另一种方式:

 

默认的任务的类实现org.quartz.Job接口,此时任务是stateless(无状态的),即会出现并行的情况,那么如何避免这种情况发生呢?

 

解决方案:使QuartzJobBean类实现org.quartz.StatefulJob接口即可(StatefulJob接口仅仅是扩展了 Job 接口,未加入新的方法,可以不需实现Job接口了,那么此时任务就会变成stateful(有状态的),此时的任务也就会串行执行了。

 

 

 

public class BackCoupon implements StatefulJob {
 
    @Override
    public void execute(JobExecutionContext context)
              throws JobExecutionException {
 
    }
 
}

注: 
在Quartz中,如果实现org.quartz.Job接口,那么这个job是stateless的,job实例的参数不能在多个任务之间共享,如果实现org.quartz.StatefulJob,这个job是个单例的,job实例的属性可以从当前任务传递到下一个任务。 
3.Spring task任务执行是并行还是串行,暂时不知。

分享到:
评论
1 楼 josh_123 2017-05-10  
讲的不错,很详细,如果quartz定时任务类采用不继承任何类的方式配置,但targemethod方法中使用多线程去执行抓取页面的任务,出现线程卡住是怎么回事,不知道楼主有没有做过类似的事情

相关推荐

    jdk-1.8.rar

    - **Stream API**:提供了一种新的数据处理方式,允许对集合进行高效的并行和串行操作,如过滤、映射和聚合。 - **日期和时间API**:Java 8用`java.time`包取代了过时的`java.util.Date`和`java.util.Calendar`,...

    jdk1.8中文.7z

    - **流(Stream)**:流API提供了一种新的数据操作方式,用于处理集合数据,支持并行和串行处理,极大提升了数据处理的效率。 - **日期和时间API的改进**:Java 8引入了全新的java.time包,取代了过时的java.util....

    面试题全集(周瑜).pdf

    * 并发、并行、串行之间的区别 * Java死锁如何避免? * 线程池的底层工作原理 * ReentrantLock中的公平锁和非公平锁的底层实现 * CountDownLatch和Semaphore的区别和底层原理 四、Java异常处理 * Java中的异常体系...

    java帮助文档,下载直接用

    5. **Stream API**:同样在Java 8中引入,提供了对集合数据进行操作的新方式,支持串行和并行处理。 6. **NIO.2**:非阻塞I/O模型,提高了网络编程的效率。 7. **反射**:允许程序在运行时检查和操作类、接口、...

    2022面试题1java背诵版本.doc

    Spring AOP通过动态代理实现,有两种方式:JDK动态代理和CGLIB代理。前者针对接口,后者针对类。 2. **HashMap底层数据结构**:HashMap基于数组+链表的数据结构,也称为拉链法。每个元素包含一个键值对,存储在一个...

    程序员面试2023,集结了阿里、腾讯、京东、美团一线大厂面试实战

    Spring通过动态代理(JDK或CGLIB)实现AOP,理解代理模式和通知类型是关键。 2. **HashMap的底层数据结构**:HashMap基于数组+链表的结构,利用哈希函数实现快速查找。理解负载因子和扩容策略对于深入理解其性能至...

    Java技术文档

    7. **多线程**:Java内置对多线程的支持,允许程序同时执行多个任务。通过继承 `Thread` 类或实现 `Runnable` 接口可以创建线程。`synchronized` 关键字用于线程同步,避免并发访问资源时的冲突。 8. **网络编程**...

    2022面试题目java背诵版本

    它的实现基于动态代理,可以是JDK动态代理或CGLIB。 2. **HashMap底层数据结构**: HashMap使用了一个Entry数组存储键值对,每个Entry包含键、值和指向下一个Entry的引用,形成链表结构。当哈希冲突发生时,元素...

    Reactor指南中文版 2.0

    异步编程指的是程序的某部分能够在等待其他部分完成时继续执行其他任务。在处理大量数据或需要快速响应的场景中,异步编程尤其重要,可以极大地提升应用的性能和吞吐量。然而,异步编程相对复杂,JVM平台上提供了...

    java开发技术

    2. Stream API:用于处理集合数据的新API,支持串行和并行操作,提供了更高效的集合处理方式。 3. 泛型方法和通配符:增强泛型功能,使代码更具灵活性和可读性。 4. 高级并发:Java提供了丰富的并发工具类,如...

    【白雪红叶】JAVA学习技术栈梳理思维导图.xmind

    jdk logger 测试框架 测试框架 junit easymock testng mockito bug管理 禅道 jira 开发工具 编程工具 eclipse myeclipse idea vi VS webstorm sublime text 版本控制 svn git 项目管理 ...

    Java

    Java 8引入的Stream API提供了处理集合的新方式,支持串行和并行操作,便于数据过滤、映射和聚合。 18. **Java内存模型(JMM)**: JMM规定了线程如何共享和访问内存,确保多线程环境下的可见性和一致性。 19. *...

Global site tag (gtag.js) - Google Analytics