spring task 定时

最近工作中需要用到定时任务的功能,虽然Spring3也自带了一个轻量级的定时任务实现,但感觉不够灵活,功能也不够强大。在考虑之后,决定整合更为专业的Quartz来实现定时任务功能。

spring task 定时

首先,当然是添加依赖的jar文件,我的项目是maven管理的,以下的我项目的依赖:

  1. <dependencies>
  2. <dependency>
  3. <groupId>org.springframework</groupId>
  4. <artifactId>spring-core</artifactId>
  5. <version>${spring.version}</version>
  6. </dependency>
  7. <dependency>
  8. <groupId>org.springframework</groupId>
  9. <artifactId>spring-context</artifactId>
  10. <version>${spring.version}</version>
  11. </dependency>
  12. <dependency>
  13. <groupId>org.springframework</groupId>
  14. <artifactId>spring-web</artifactId>
  15. <version>${spring.version}</version>
  16. </dependency>
  17. <dependency>
  18. <groupId>org.springframework</groupId>
  19. <artifactId>spring-tx</artifactId>
  20. <version>${spring.version}</version>
  21. </dependency>
  22. <dependency>
  23. <groupId>org.springframework</groupId>
  24. <artifactId>spring-jdbc</artifactId>
  25. <version>${spring.version}</version>
  26. </dependency>
  27. <dependency>
  28. <groupId>org.mybatis</groupId>
  29. <artifactId>mybatis</artifactId>
  30. <version>${mybatis.version}</version>
  31. </dependency>
  32. <dependency>
  33. <groupId>org.aspectj</groupId>
  34. <artifactId>aspectjweaver</artifactId>
  35. <version>1.7.4</version>
  36. </dependency>
  37. <dependency>
  38. <groupId>org.mybatis</groupId>
  39. <artifactId>mybatis-spring</artifactId>
  40. <version>${mybatis.spring.version}</version>
  41. </dependency>
  42. <dependency>
  43. <groupId>org.slf4j</groupId>
  44. <artifactId>slf4j-log4j12</artifactId>
  45. <version>${slf4j.version}</version>
  46. </dependency>
  47. <dependency>
  48. <groupId>commons-lang</groupId>
  49. <artifactId>commons-lang</artifactId>
  50. <version>${commons.lang.version}</version>
  51. </dependency>
  52. <dependency>
  53. <groupId>commons-dbcp</groupId>
  54. <artifactId>commons-dbcp</artifactId>
  55. <version>${commons.dbcp.version}</version>
  56. </dependency>
  57. <dependency>
  58. <groupId>com.oracle</groupId>
  59. <artifactId>ojdbc14</artifactId>
  60. <version>${ojdbc.version}</version>
  61. </dependency>
  62. <dependency>
  63. <groupId>org.springframework</groupId>
  64. <artifactId>spring-context-support</artifactId>
  65. <version>${spring.version}</version>
  66. </dependency>
  67. <dependency>
  68. <groupId>org.quartz-scheduler</groupId>
  69. <artifactId>quartz</artifactId>
  70. <version>${quartz.version}</version>
  71. </dependency>
  72. </dependencies>

或许你应该看出来了,我的项目是spring整合了mybatis,目前spring的最新版本已经到了4.x系列,但是最新版的mybatis-spring的整合插件所依赖推荐的依然是spring 3.1.3.RELEASE,所以这里没有用spring的最新版而是用了推荐的3.1.3.RELEASE,毕竟最新版本的功能一般情况下也用不到。

至于quartz,则是用了目前的最新版2.2.1

之所以在这里特别对版本作一下说明,是因为spring和quartz的整合对版本是有要求的。

spring3.1以下的版本必须使用quartz1.x系列,3.1以上的版本才支持quartz 2.x,不然会出错。

至于原因,则是spring对于quartz的支持实现,org.springframework.scheduling.quartz.CronTriggerBean继承了org.quartz.CronTrigger,在quartz1.x系列中org.quartz.CronTrigger是个类,而在quartz2.x系列中org.quartz.CronTrigger变成了接口,从而造成无法用spring的方式配置quartz的触发器(trigger)。

在Spring中使用Quartz有两种方式实现:第一种是任务类继承QuartzJobBean,第二种则是在配置文件里定义任务类和要执行的方法,类和方法可以是普通类。很显然,第二种方式远比第一种方式来的灵活。

这里采用的就是第二种方式。

spring配置文件:

  1. <!-- 使用MethodInvokingJobDetailFactoryBean,任务类可以不实现Job接口,通过targetMethod指定调用方法-->
  2. <bean id="taskJob" class="com.tyyd.dw.task.DataConversionTask"/>
  3. <bean id="jobDetail" class="org.springframework.scheduling.quartz.MethodInvokingJobDetailFactoryBean">
  4. <property name="group" value="job_work"/>
  5. <property name="name" value="job_work_name"/>
  6. <!--false表示等上一个任务执行完后再开启新的任务-->
  7. <property name="concurrent" value="false"/>
  8. <property name="targetObject">
  9. <ref bean="taskJob"/>
  10. </property>
  11. <property name="targetMethod">
  12. <value>run</value>
  13. </property>
  14. </bean>
  15. <!-- 调度触发器 -->
  16. <bean id="myTrigger"
  17. class="org.springframework.scheduling.quartz.CronTriggerFactoryBean">
  18. <property name="name" value="work_default_name"/>
  19. <property name="group" value="work_default"/>
  20. <property name="jobDetail">
  21. <ref bean="jobDetail" />
  22. </property>
  23. <property name="cronExpression">
  24. <value>0/5 * * * * ?</value>
  25. </property>
  26. </bean>
  27. <!-- 调度工厂 -->
  28. <bean id="scheduler" class="org.springframework.scheduling.quartz.SchedulerFactoryBean">
  29. <property name="triggers">
  30. <list>
  31. <ref bean="myTrigger"/>
  32. </list>
  33. </property>
  34. </bean>

Task类则是一个普通的Java类,没有继承任何类和实现任何接口(当然可以用注解方式来声明bean):

  1. //@Component
  2. public class DataConversionTask{
  3. /** 日志对象 */
  4. private static final Logger LOG = LoggerFactory.getLogger(DataConversionTask.class);
  5. public void run() {
  6. if (LOG.isInfoEnabled()) {
  7. LOG.info("数据转换任务线程开始执行");
  8. }
  9. }
  10. }

至此,简单的整合大功告成,run方法将每隔5秒执行一次,因为配置了concurrent等于false,所以假如run方法的执行时间超过5秒,在执行完之前即使时间已经超过了5秒下一个定时计划执行任务仍不会被开启,如果是true,则不管是否执行完,时间到了都将开启。

接下去,将实现如何动态的修改定时执行的时间,以及如何停止正在执行的任务,待续,,,

顺便贴一下cronExpression表达式备忘: 字段 允许值 允许的特殊字符

秒 0-59 , – * /

分 0-59 , – * /

小时 0-23 , – * /

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

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

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

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

表达式意义

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

每天早上6点

0 6 * * *

每两个小时

0 */2 * * *

晚上11点到早上8点之间每两个小时,早上八点

0 23-7/2,8 * * *

每个月的4号和每个礼拜的礼拜一到礼拜三的早上11点

0 11 4 * 1-3

1月1日早上4点

0 4 1 1 *

转至:http://www.meiriyouke.net/?p=82

上一篇:javascript声明对象时 带var和不带var的区别


下一篇:如何快速定位找出SEGV内存错误的程序Bug