1. spring学习系列 -- 定时器一TimerTask

spring定时器一般有两种

TimerTask、Quartz。本节只讲TimerTask

需要的包

aopalliance-1.0.jar

commons-logging-1.1.1.jar

spring-aop-3.0.6.RELEASE.jar

spring-asm-3.0.6.RELEASE.jar

spring-beans-3.0.6.RELEASE.jar

spring-context-3.0.6.RELEASE.jar

spring-core-3.0.6.RELEASE.jar

spring-expression-3.0.6.RELEASE.jar

TimerTask实例:同时启动2个定时器,执行任务

定时执行任务的类继承TimerTask:

Java代码

  1. public class EmailReportTask extends TimerTask{
  2. @Override
  3. public void run() {
  4. System.out.println(" EmailReportTask Run... ");
  5. }
  6. }
  7. public class PageReportTask extends TimerTask{
  8. @Override
  9. public void run() {
  10. System.out.println("PageReportTask Run...");
  11. }
  12. }

spring的配置文件:

Xml代码

  1. <!--  Bean  -->
  2. <bean id="emailReportTask" class="com.hry.spring.timertask.EmailReportTask" />
  3. <bean id="pageReportTask" class="com.hry.spring.timertask.PageReportTask" />
  4. <!-- ScheduledTimerTask设置定时器属性 : period=定时器周期;delay=延迟多久启动
  5. 86400000代表24个小时;timerTask=执行定时任务的类对象  -->
  6. <bean id="emailReportScheduleReportTask"
  7. class="org.springframework.scheduling.timer.ScheduledTimerTask">
  8. <property name="timerTask" ref="emailReportTask" />
  9. <property name="period" value="2000" />
  10. <property name="delay" value="1000" />
  11. </bean>
  12. <bean id="pageReportScheduleReportTask"
  13. class="org.springframework.scheduling.timer.ScheduledTimerTask">
  14. <property name="timerTask" ref="pageReportTask" />
  15. <property name="period" value="2000" />
  16. </bean>
  17. <!-- Spring的TimerFactoryBean负责启动定时任务;
  18. scheduledTimerTasks = 需要启动的定时器任务的列表-->
  19. <bean class="org.springframework.scheduling.timer.TimerFactoryBean">
  20. <property name="scheduledTimerTasks">
  21. <list>
  22. <ref bean="emailReportScheduleReportTask"/>
  23. <ref bean="pageReportScheduleReportTask"/>
  24. </list>
  25. </property>
  26. </bean>

测试代码:

Java代码

  1. public class TestBaseService {
  2. protected ApplicationContext ctx = new ClassPathXmlApplicationContext(
  3. new String[]{
  4. "classpath:resource/spring.xml"
  5. }
  6. );
  7. @Test
  8. public void timer(){
  9. try{
  10. // 这个是主线程,如果结束了,则定时器也会结束,所有设置时间要长
  11. Thread.sleep(36 * 1000);
  12. }catch(Exception e){
  13. e.printStackTrace();
  14. }
  15. }
  16. }

参考文献:

  • spring 定时器如何配置
  • Spring中Quartz的配置
  • spring定时任务之quartz
  • Spring定时器的两种实现方式

2.Spring的第二种定时器quartz

在上一节的基础上,讨论Spring的第二种定时器quartz

重点包:

quartz-1.8.6.jar

org.springframework.context.support-3.1.1.RELEASE.jar

  包说明

quartz包请使用1.8.6或以下的版本,因为quartz2.0版本和spring3.1.1存在冲突,会抛出

目的

通过quartz定时循环执行一个任务

任务类

该类还包含一个属性sTest及其set/get方法

Java代码

  1. public class MyJob extends QuartzJobBean {
  2. private String sTest;
  3. @Override
  4. protected void executeInternal(JobExecutionContext job)
  5. throws JobExecutionException {
  6. // TODO Auto-generated method stub
  7. System.out.println("sTest = " + sTest);
  8. System.out.println("MyJob Run...");
  9. }
  10. public String getsTest() {
  11. return sTest;
  12. }
  13. public void setsTest(String sTest) {
  14. this.sTest = sTest;
  15. }
  16. }

spring的配置文件

JobDetailBean:设置要执行任务的类,在这里通过jobDataAsMap属性还可以设置任务类的属性。

Trigger用于设置人物类启动的时间,循环的间隔等工作信息。spring包含2种Trigger :

simpleReportTrigger功能类似上一篇文章的TimerTask,设置任务延迟多久启动,循环间隔等信息

cronReportTrigger可以设置任务精确工作的时间

SchedulerFactoryBean:仅仅有以上2个类设置是不够的,如果要启动定时器,还需要通过此类设置要启动的trigger。

Xml代码

  1. <!--
  2. JobDetailBean是Quartz的org.quartz.JobDetail的子类,它要求通过jobClass属性来设置一个Job对象。
  3. jobClass = 要执行定时任务的类
  4. jobDataAsMap = 用于向任务类对象中注入信息,即可以注入值,也可以引用另一个变量
  5. -->
  6. <bean id="reportJob" class="org.springframework.scheduling.quartz.JobDetailBean">
  7. <property name="jobClass" >
  8. <value>com.hry.spring.timertask.MyJob</value>
  9. </property>
  10. <property name="jobDataAsMap">
  11. <map>
  12. <entry key="sTest">
  13. <value>10</value>
  14. </entry>
  15. <!--
  16. <entry key="courseService">
  17. <ref bean="courseService"/>
  18. </entry>
  19. -->
  20. </map>
  21. </property>
  22. </bean>
  23. <!--
  24. org.quartz.Trigger用于设置任务类如何工作
  25. SimpleTriggerBean 类似 ScheduledTimerTask,配置定时器的执行频率和延迟多久执行
  26. jobDetail = 实际工作类
  27. startDelay = 延迟实际
  28. repeatInterval = 重复频率
  29. -->
  30. <bean id="simpleReportTrigger" class="org.springframework.scheduling.quartz.SimpleTriggerBean">
  31. <property name="jobDetail" ref="reportJob" />
  32. <property name="startDelay" value="1000" />
  33. <property name="repeatInterval" value="2000" />
  34. </bean>
  35. <!--
  36. CronTriggerBean 指定某个时间允许任务
  37. cronExpression = 通过表达式设置特定的时间点执行
  38. -->
  39. <bean id="cronReportTrigger" class="org.springframework.scheduling.quartz.CronTriggerBean">
  40. <property name="jobDetail" ref="reportJob" />
  41. <property name="cronExpression" value = "40 * * * * ?" />
  42. </bean>
  43. <!--
  44. SchedulerFactoryBean:启动定时器
  45. triggers = 启动哪些定时任务,可以多个
  46. -->
  47. <bean class="org.springframework.scheduling.quartz.SchedulerFactoryBean">
  48. <property name="triggers">
  49. <list>
  50. <ref bean="simpleReportTrigger"/>
  51. <!--
  52. <ref bean="cronReportTrigger" />
  53. -->
  54. </list>
  55. </property>
  56. </bean>

通过上篇文章的测试代码执行以上程序,则有如下输入,说明我们程序成功了。

Html代码

  1. sTest = 10
  2. MyJob Run...
  3. sTest = 10
  4. MyJob Run...
  5. sTest = 10
  6. MyJob Run...
  7. sTest = 10
  8. MyJob Run...
  9. sTest = 10
  10. MyJob Run...

CronTriggerBean的精华是cron表达式,以下是关于cron表达式(来自网络)

Cron 表达式依照顺序有7 个字段:

小时

月内日期

周内日期

年(可选字段)

特殊字符

Cron 触发器利用一系列特殊字符,如下所示:

反斜线(/)字符表示增量值。例如,在秒字段中“5/15”代表从第 5 秒开始,每 15 秒一次。

问号(?)字符和字母 L 字符只有在月内日期和周内日期字段中可用。问号表示这个字段不包含具体值。所以,如果指定月内日期,可以在周内日期字段中插入“?”,表示周内日期值无关紧要。字母 L 字符是 last 的缩写。放在月内日期字段中,表示安排在当月最后一天执行。在周内日期字段中,如果“L”单独存在,就等于“7”,否则代表当月内周内日期的最后一个实例。所以“0L”表示安排在当月的最后一个星期日执行。

在月内日期字段中的字母(W)字符把执行安排在最靠近指定值的工作日。把“1W”放在月内日期字段中,表示把执行安排在当月的第一个工作日内。

井号(#)字符为给定月份指定具体的工作日实例。把“MON#2”放在周内日期字段中,表示把任务安排在当月的第二个星期一。

星号(*)字符是通配字符,表示该字段可以接受任何可能的值。

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

秒 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 *

3.Spring的两种定时器的区别

前2节绍了spring的两种定时器:TimerTask 和 Quartz,本节要在这两节的基础上,讲讲两者的区别。

精确度和功能

Quartz可以通过cron表达式精确到特定时间执行,而TimerTask不能。Quartz拥有TimerTask所有的功能,而TimerTask则没有。

任务类的数量

TimerTask和Quartz每次执行任务时,每次调用的是不是都是同一个任务类对象,还是每次都不一样?现在做如下实验,每次执行任务时,将任务类对象本身打印出来。

Quartz任务类

Java代码

  1. public class MyJob extends QuartzJobBean {
  2. private String sTest;
  3. @Override
  4. protected void executeInternal(JobExecutionContext job)
  5. throws JobExecutionException {
  6. // TODO Auto-generated method stub
  7. System.out.println("sTest = " + sTest);
  8. System.out.println("MyJob Run..." + this);
  9. }
  10. // set/get 略
  11. }

Quartz输出结果

Java代码

  1. sTest = 10
  2. MyJob Run...com.hry.spring.timertask.MyJob@1060478
  3. sTest = 10
  4. MyJob Run...com.hry.spring.timertask.MyJob@db4fa2
  5. sTest = 10
  6. MyJob Run...com.hry.spring.timertask.MyJob@491c4c

从输出结果可以看出,Quartz每次执行都创建一个新的任务类对象。

TimerTask任务类

Java代码

  1. public class EmailReportTask extends TimerTask{
  2. // 每次执行过程中num的值都会发生变化,说明此事使用的是同一个类对象
  3. private int num = 0;
  4. @Override
  5. public void run() {
  6. System.out.println("num = " + num++);
  7. System.out.println(this);
  8. }
  9. }

TimerTask任务类的输出结果

Java代码

  1. num = 0
  2. com.hry.spring.timertask.EmailReportTask@1581593
  3. PageReportTask Run...
  4. num = 1
  5. com.hry.spring.timertask.EmailReportTask@1581593
  6. PageReportTask Run...

从输出结果可以看出,TimerTask每次执行时,都是使用同一个对象

从以上的分析,可以得出结论:Quartz每次执行任务都创建一个新的任务类对象,而TimerTask则每次使用同一个任务类对象。

对异常的处理

一个循环执行的任务,如果某一次执行任务时,因为某些原因抛出异常,则定时器是否还会在下一个执行任务的时间点执行任务吗?下面通过模拟在任务类中抛出异常,来模拟这种情况,并测试两种定时器如何处理这种情况。

Quartz任务类

Java代码

  1. public class MyJob extends QuartzJobBean {
  2. private String sTest;
  3. @Override
  4. protected void executeInternal(JobExecutionContext job)
  5. throws JobExecutionException {
  6. // TODO Auto-generated method stub
  7. System.out.println("sTest = " + sTest);
  8. System.out.println("MyJob Run..." + this);
  9. throw new RuntimeException("Test");
  10. }
  11. // set/get 方法略
  12. }

Quartz输出结果

Html代码

  1. sTest = 10
  2. MyJob Run...com.hry.spring.timertask.MyJob@16f25a7
  3. 2013-01-05 19:58:37,381 [org.springframework.scheduling.quartz.SchedulerFactoryBean#0_Worker-3] ERROR [org.quartz.core.JobRunShell] - Job DEFAULT.reportJob threw an unhandled Exception:
  4. java.lang.RuntimeException: Test
  5. at com.hry.spring.timertask.MyJob.executeInternal(MyJob.java:24)
  6. at org.springframework.scheduling.quartz.QuartzJobBean.execute(QuartzJobBean.java:113)
  7. at org.quartz.core.JobRunShell.run(JobRunShell.java:223)
  8. at org.quartz.simpl.SimpleThreadPool$WorkerThread.run(SimpleThreadPool.java:549)
  9. 2013-01-05 19:58:37,396 [org.springframework.scheduling.quartz.SchedulerFactoryBean#0_Worker-3] ERROR [org.quartz.core.ErrorLogger] - Job (DEFAULT.reportJob threw an exception.
  10. org.quartz.SchedulerException: Job threw an unhandled exception. [See nested exception: java.lang.RuntimeException: Test]
  11. at org.quartz.core.JobRunShell.run(JobRunShell.java:234)
  12. at org.quartz.simpl.SimpleThreadPool$WorkerThread.run(SimpleThreadPool.java:549)
  13. Caused by: java.lang.RuntimeException: Test
  14. at com.hry.spring.timertask.MyJob.executeInternal(MyJob.java:24)
  15. at org.springframework.scheduling.quartz.QuartzJobBean.execute(QuartzJobBean.java:113)
  16. at org.quartz.core.JobRunShell.run(JobRunShell.java:223)
  17. ... 1 more
  18. sTest = 10
  19. MyJob Run...com.hry.spring.timertask.MyJob@110c31
  20. 2013-01-05 19:58:39,381 [org.springframework.scheduling.quartz.SchedulerFactoryBean#0_Worker-4] ERROR [org.quartz.core.JobRunShell] - Job DEFAULT.reportJob threw an unhandled Exception:
  21. java.lang.RuntimeException: Test
  22. at com.hry.spring.timertask.MyJob.executeInternal(MyJob.java:24)
  23. at org.springframework.scheduling.quartz.QuartzJobBean.execute(QuartzJobBean.java:113)
  24. at org.quartz.core.JobRunShell.run(JobRunShell.java:223)
  25. at org.quartz.simpl.SimpleThreadPool$WorkerThread.run(SimpleThreadPool.java:549)
  26. 2013-01-05 19:58:39,396 [org.springframework.scheduling.quartz.SchedulerFactoryBean#0_Worker-4] ERROR [org.quartz.core.ErrorLogger] - Job (DEFAULT.reportJob threw an exception.
  27. org.quartz.SchedulerException: Job threw an unhandled exception. [See nested exception: java.lang.RuntimeException: Test]
  28. at org.quartz.core.JobRunShell.run(JobRunShell.java:234)
  29. at org.quartz.simpl.SimpleThreadPool$WorkerThread.run(SimpleThreadPool.java:549)
  30. Caused by: java.lang.RuntimeException: Test
  31. at com.hry.spring.timertask.MyJob.executeInternal(MyJob.java:24)
  32. at org.springframework.scheduling.quartz.QuartzJobBean.execute(QuartzJobBean.java:113)
  33. at org.quartz.core.JobRunShell.run(JobRunShell.java:223)
  34. ... 1 more

以上结果中多次出现

Java代码

  1. sTest = 10
  2. MyJob Run...com.hry.spring.timertask.MyJob@110c31

可以看出,尽管每次执行任务时,任务类都会抛出异常,但是Quartz定时器,依然在下一个任务执行时间点执行任务,并没有因为异常,而导致定时器关闭,不再执行循环任务。

TimerTask任务类

Java代码

  1. public class EmailReportTask extends TimerTask{
  2. // 每次执行过程中num的值都会发生变化,说明此事使用的是同一个类对象
  3. private int num = 0;
  4. @Override
  5. public void run() {
  6. System.out.println("num = " + num++);
  7. throw new RuntimeException("test");
  8. }
  9. }

TimerTask输出结果

Java代码

  1. num = 0
  2. Exception in thread "org.springframework.scheduling.timer.TimerFactoryBean#0" java.lang.RuntimeException: test
  3. at com.hry.spring.timertask.EmailReportTask.run(EmailReportTask.java:11)
  4. at java.util.TimerThread.mainLoop(Unknown Source)
  5. at java.util.TimerThread.run(Unknown Source)

TimerTask抛出异常后,后续再也没有执行此任务了,并且定时器所在的线程也自动结束。

通过以上的分析,可以知道Quartz的某次执行任务过程中抛出异常,不影响下一次任务的执行,当下一次执行时间到来时,定时器会再次执行任务;而TimerTask则不同,一旦某个任务在执行过程中抛出异常,则整个定时器生命周期就结束,以后永远不会再执行定时器任务。

Spring的两种定时器相关推荐

  1. Spring AOP两种实现机制是什么?

    Spring AOP两种实现机制是什么? 1.如果是有接口声明的类进行AOP 时,spring调用的是java.lang.reflection.Proxy 类来做处理 2.如果是没有接口声明的类时, ...

  2. Java框架篇---spring aop两种配置方式

    Java框架篇---spring aop两种配置方式 第一种:注解配置AOP 注解配置AOP(使用 AspectJ 类库实现的),大致分为三步:  1. 使用注解@Aspect来定义一个切面,在切面中 ...

  3. Spring中两种编程式事务管理

    Spring中两种编程式事务管理 在代码中显示调用beginTransaction,commit,rollback等与事务处理相关的方法,这就是编程式事务管理,当只有少数事务操作时,编程式事务管理才比 ...

  4. js中的两种定时器setTimeout()和setInterval()怎么用

    JS有两种定时器分别是setTimeout()和setInterval(),这两个区别就是setTimeout()是一次性的定时器,而setInterval()是循环的定时器. 定时器的精度 js中的 ...

  5. Qt中两种定时器用法

    在Qt中使用定时器有两种方法,一种是使用QObiect类的定时器:一种是使用QTimer类.定时器的精确性依赖于操作系统和硬件,大多数平台支持20ms的精确度. 1.QObject类的定时器 QObj ...

  6. Linux两种定时器

    Linux下的定时器有两种,以下分别介绍: 1.alarm 如果不要求很精确的话,用alarm()和signal()就够了 unsigned int alarm(unsigned int second ...

  7. Spring的两种任务调度Scheduled和Async

    Spring提供了两种后台任务的方法,分别是: 调度任务,@Schedule 异步任务,@Async 当然,使用这两个是有条件的,需要在spring应用的上下文中声明 <task:annotat ...

  8. Spring AOP两种使用方式以及如何使用解析

    AOP是一种面向切面编程思想,也是面向对象设计(OOP)的一种延伸. 在Spring实现AOP有两种实现方式,一种是采用JDK动态代理实现,另外一种就是采用CGLIB代理实现,Spring是如何实现的 ...

  9. Spring的两种代理方式:JDK动态代理和CGLIB动态代理

    代理模式 代理模式的英文叫做Proxy或Surrogate,中文都可译为"代理",所谓代理,就是一个人或者一个机构代表另一个人或者另一个机构采取行动.在一些情况下,一个客户不想或者 ...

  10. js中两种定时器,setTimeout和setInterval的区别

    2019独角兽企业重金招聘Python工程师标准>>> setTimeout只在指定时间后执行一次,代码如下: <script> //定时器 异步运行 function ...

最新文章

  1. share一下一线大厂是怎么招聘技术岗的?
  2. 黄海广博士的机器学习个人笔记及深度学习笔记
  3. python利器下载-Python打包利器:auto-py-to-exe
  4. 你真的会搜索?低效的你简直在浪费生命(三)(终结篇)
  5. CentOS 7配置静态IP地址 解决了IP失效问题
  6. 关于iOS原生条形码扫描,你需要注意的两三事
  7. C++中的临时对象都是const类型
  8. 连接SQL SERVER的时候登录名如何清除
  9. Java基于opencv实现图像数字识别(二)—基本流程
  10. 小型水下机器人控制系统方案设计
  11. 【Opencv小项目 1】Opencv实现简单颜色识别
  12. C1实训-动态令牌(二次验证码)原理及实现
  13. 怎么删除电脑服务器远程桌面连接,删除远程桌面服务客户端访问许可证
  14. 与通用计算机相比较 单片机优势在哪,单片机原理与嵌入式系统-中国大学mooc-题库零氪...
  15. 淘宝哪些退款原因会影响店铺?具体介绍
  16. 测绘资质通用标准审查细则
  17. JS 下载文件方法分享(解决图片文件无法直接下载和 IE兼容问题)
  18. python提取视频、音频音轨,并合成到视频中
  19. 2021/3/6 OJ每日一题 小媛在努力
  20. 北京2008奥运会门票到手

热门文章

  1. hsqldb和mysql,Hsqldb简介和基本使用
  2. c语言连连看实验报告,连连看c语言实验报告.docx
  3. 织梦的网站地图怎么做html,织梦自带网站地图的生成制作方法详解
  4. 苹果carplay下载_苹果宣布推出CarPlay
  5. java如何准确的读取多音字
  6. App测试实战:测试内容、测试工具、测试效果
  7. Dotween Yoyo
  8. HTML5轮播图全代码
  9. 记一次阿里云RDS MYSQL 数据恢复的经历
  10. vue3 three ts 全景图