Spring+Quartz实现定时任务的几种方式

 

Spring整合Quartz实现定时任务步骤很简单,大致需要经过如下几步:

创建任务(Job)、配置JobDetail、配置触发器(Trigger)、配置SchedulerFactoryBean

首先使用Maven创建一个web项目并引入Spring和quartz的依赖(习惯使用这种开发模式了,当然不使用Maven也可以)

需要引入的依赖包如下:

spring定时任务setname Spring定时任务job_定时任务

此处我们使用的是Spring3.2.6+Quartz2.2.1版本进行实验。

增加完Spring和Quartz的依赖包之后,就需要创建Spring配置文件,并在web.xml文件中引入Spring支持。

spring定时任务setname Spring定时任务job_触发器_02

好了,到现在为止,基本的环境算是搭建好了,接下来我们就要开始Spring+Quartz实现定时任务了。

一、创建任务(Job)

Spring+Quartz实现Job有两种方式:

第一种是继承org.springframework.scheduling.quartz.QuartzJobBean类来实现Job任务,并实现里面的抽象方法executeInternal;另一种是不继承任何类,创建普通的Java类,然后自己指定任务的执行方法(个人感觉此种方式较好,实现起来方便而且大大降低了系统的业务的耦合性)。

 继承QuartzJobBean类实现定时

我们先来看一下继承QuartzJobBean类的这种形式,创建一个任务类ExampleJob,具体代码如下:

spring定时任务setname Spring定时任务job_定时任务_03

为了演示方便,executeInternal方法中没有写复杂的业务逻辑,只简单的输出一句话,真正的生产环境中在该方法中实现你所需要的业务逻辑即可。

二、在Spring配置文件中配置JobDetail

 

spring定时任务setname Spring定时任务job_spring定时任务setname_04

三、配置触发器(Trigger)

Spring提供了两种触发器,如下:

1、org.springframework.scheduling.quartz.SimpleTriggerFactoryBean(此种方式是隔多长时间进行触发一次,比如每隔24小时触发一次)

2、org.springframework.scheduling.quartz.CronTriggerFactoryBean(此种方式是在指定的时间进行触发,比如只在周一进行触发。不过根据配置也很方便的实现类似SimpleTriggerFactoryBean形式的定时任务)

Spring所提供的这两种触发器方式和前面提到的任务创建方式均可以相互之间混用,很灵活。

这里我们先使用SimpleTriggerFactoryBean这个trigger来配置

spring定时任务setname Spring定时任务job_字段_05

四、配置SchedulerFactoryBean

 

spring定时任务setname Spring定时任务job_spring定时任务setname_06

好了,到现在为止,一个简单的定时任务就完成了,下面我们来启动一下web项目,看运行结果如何。

 

spring定时任务setname Spring定时任务job_定时任务_07

很不幸的是运行失败了,这是为啥呢?别着急,慢慢看看异常信息(学会分析异常信息也是程序员重要的能力之一噢)。从异常信息中我们可以看出,JobDetailBean引用了一个接口来作为父类了。既然是这样的话,那我们就需要看一下JobDetailBean的源码了。

spring定时任务setname Spring定时任务job_spring定时任务setname_08

从源码中我们可以看到Spring的JobDetailBean继承了Quartz的JobDetail,接下来我们就要看Quartz的JobDetail这个源码了。

 

spring定时任务setname Spring定时任务job_spring定时任务setname_09

在Quartz中JobDetail竟然是一个接口。是不是搞错了,你肯定要问了。这是为什么呢,为什么呢,为什么呢?哈哈,这个就得谈到Quartz的历史问题了。Quartz从1.X升级到2.X之后,JobDetail由类修改为接口了,为啥要改?那你得去问Quartz作者了,嘿嘿。那如果再继续采用这种模式的话那肯定会错喽。那我们对它就没有法子了么?怎么可能。处理它很简单啊:一、把Quartz降到1.X版本;二、更改Job的实现方式。

首先我们来试试使用1.X版本的Quartz

 

spring定时任务setname Spring定时任务job_字段_10

spring定时任务setname Spring定时任务job_定时任务_11

可以看到在Quartz1.X版本中,JobDetail还是一个类。接下来我们再启动一下web项目看看运行结果何如。

 

spring定时任务setname Spring定时任务job_定时任务_12

可以看到,在更换Quartz版本为1.X之后,定时任务正常运行了。

看到这,想必大家就要问了,其他方式呢,总不能让我们一直使用Quartz1.X版本吧?当然不是,我们还有很多办法,大家慢慢往下看。

3.2 创建普通的Java类实现定时

首先创建一个基本的Java类来做为Job任务类,代码如下:

spring定时任务setname Spring定时任务job_触发器_13

接下来是在Spring配置文件中配置JobDetail、Trigger、SchedulerFactoryBean

 

spring定时任务setname Spring定时任务job_触发器_14

好了,配置完之后我们再启动一下程序,看看定时任务时否运行良好(这里我们使用的Trigger是CronTriggerFactoryBean,当然也可以使用SimpleTriggerFactoryBean)

关于Trigger中时间如何配置,quartz官网描述的很清楚,大家可以参考如下网址:http://quartz-scheduler.org/documentation/quartz-2.2.x/tutorials/crontrigger

 

spring定时任务setname Spring定时任务job_spring定时任务setname_15

OK,没有问题,好了,关于Spring+Quartz实现定时任务暂就告一段落,欢迎大家多多交流,有不当之后还请大家指正出来。

本示例对应的代码下载地址为:

3.3 Quartz定时器的时间设置

时间的配置如下:<value>0 26 16 * * ?</value>

这些星号由左到右按顺序代表 :

 

序号

1

2

3

4

5

6

7

格式

 

序号

说明

是否必填

允许填写的值

允许的通配符

1

0-59

,-*/

2

0-59

,-*/

3

0-23

,-*/

4

1-31

,-*/LW

5

1-12 or JAN-DEC

,-*/

6

1-7 or SUN-STA

,-*/L#

7

empty or 1970-2099

,-*/

通配符说明:

通配符

说明

*

 

表示所有值. 例如:在分的字段上设置“*”,表示每一分钟都会触发。

 

?

表示不指定值。使用的场景为不需要关心当前设置这个字段的值。例如:要在每月的10号触发一个操作,但不关心是周几,所以需要周位置的那个字段设置为"?" 具体设置为

 0 0 0 10 * ?

-

 

表示区间。例如 在小时上设置 "10-12",表示 10,11,12点都会触发。

 

,

 

表示指定多个值,例如在周字段上设置 "MON,WED,FRI" 表示周一,周三和周五触发

 

/

 

用于递增触发。如在秒上面设置"5/15" 表示从5秒开始,每增15秒触发(5,20,35,50)。在日字段上设置'1/3'所示每月1号开始,每隔三天触发一次。

 

L

表示最后的意思。在日字段设置上,表示当月的最后一天(依据当前月份,如果是二月还会依据是否是润年[leap]), 在周字段上表示星期六,相当于"7"或"SAT"。如果在"L"前加上数字,则表示该数据的最后一个。例如在周字段上设置"6L"这样的格式,则表示“本月最后一个星期五"

'L'和 'W'可以组合使用。如果在日字段上设置"LW",则表示在本月的最后一个工作日触发(一般指发工资 )

W

表示离指定日期的最近那个工作日(周一至周五). 例如在日字段上设置"15W",表示离每月15号最近的那个工作日触发。如果15号正好是周六,则找最近的周五(14号)触发, 如果15号是周未,则找最近的下周一(16号)触发.如果15号正好在工作日(周一至周五),则就在该天触发。如果指定格式为 "1W",它则表示每月1号往后最近的工作日触发。如果1号正是周六,则将在3号下周一触发。(注,"W"前只能设置具体的数字,不允许区间"-").

#

 

序号(表示每月的第几个周几),例如在周字段上设置"6#3"表示在每月的第三个周六.注意如果指定"#5",正好第五周没有周六,则不会触发该配置(用在母亲节和父亲节再合适不过了)

 

周字段的设置,若使用英文字母是不区分大小写的,MON 与mon相同

具体时间设定可参考

"0/10 * * * * ?" 每10秒触发

"0 0 12 * * ?" 每天中午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期间的每1分钟触发

"0 0/5 14 * * ?" 在每天下午2点到下午2:55期间的每5分钟触发

"0 0/5 14,18 * * ?" 在每天下午2点到2:55期间和下午6点到6:55期间的每5分钟触发

"0 0-5 14 * * ?" 在每天下午2点到下午2:05期间的每1分钟触发

"0 10,44 14 ? 3 WED" 每年三月的星期三的下午2:10和2:44触发

"0 15 10 ? * MON-FRI" 周一至周五的上午10:15触发

"0 15 10 15 * ?" 每月15日上午10:15触发

"0 15 10 L * ?" 每月最后一日的上午10:15触发

"0 15 10 ? * 6L" 每月的最后一个星期五上午10:15触发

"0 15 10 ? * 6L 2002-2005" 2002年至2005年的每月的最后一个星期五上午10:15触发

"0 15 10 ? * 6#3" 每月的第三个星期五上午10:15触发

"0 0 06,18 * * ?"  在每天上午6点和下午6点触发

"0 30 5 * * ? *"   在每天上午5:30触发

"0 0/3 * * * ?"    每3分钟触发