J'KYO
No pains,no gains.

referenc:https://blog.csdn.net/clementad/article/details/42042111

下面介绍几种常用的定时器及其实现方法:

 

第一种:Timer和TimerTask

Timer实际上是个线程,它可以定时调度一个TimerTask对象。一个TimerTask实际上就是一个拥有run方法的类,需要定时执行的代码放到run方法体内。 

Timer及其方法:

 

[java] view plain copy
 
  1. java.util.Timer timer = new java.util.Timer(true); // true表示这个timer以daemon方式运行(优先级低,程序结束或其他用户线程都结束时timer也自动结束)  
  2.   
  3. TimerTask task = new TimerTask() {   
  4.     public void run() {  
  5.      //每次需要执行的代码   
  6.     }  
  7. };    
  8.   
  9. //以下是几种调度task的方法:     
  10. timer.schedule(task, time); // time为Date类型:在指定时间执行一次。  
  11. timer.schedule(task, firstTime, period); // firstTime为Date类型,period为long,从firstTime时刻开始,每隔period毫秒执行一次。  
  12. timer.schedule(task, delay) // delay 为long类型:从现在起过delay毫秒执行一次</span>  
  13. timer.schedule(task, delay, period) // delay为long,period为long:从现在起过delay毫秒以后,每隔period毫秒执行一次。    
  14.   
  15. //下面这两种方法的参数和前面两种一样,不同的是它们可以把执行间隔受到其他任务的影响降到最低。如果由于任何原因(如垃圾回收或其他背景活动)而延迟了某次执行,则将快速连续地出现两次或更多的执行,从而使后续执行能够“追赶上来”,从而平均执行间隔极大地逼近period。  
  16. timer.scheduleAtFixedRate(task, delay, period);  
  17. timer.scheduleAtFixedRate(task, firstTime, period);  

 


一个完整的例子:

 

[java] view plain copy
 
  1. import java.util.Date;  
  2. import java.util.TimerTask;  
  3.   
  4. public class MyTimerTask extends TimerTask {  
  5.     @Override  
  6.     public void run() {  
  7.         System.out.println("定时器开始执行任务……" + new Date());  
  8.     }  
  9. }  

[java] view plain copy
 
  1. import java.util.Timer;  
  2.   
  3. public class MyTimer {  
  4.     public static void main(String[] args) {  
  5.         Timer timer = new Timer();  
  6.         int delay = 2*1000; //2秒后开始  
  7.         int period = 3*1000; //每3秒执行一次  
  8.         timer.scheduleAtFixedRate(new MyTimerTask(), delay, period);  
  9.     }  
  10. }  

第二种:使用Spring的任务调度类ScheduledExecutorTask

 

实现方法:

第一步:建立一个Java类,并实现Runnable接口:

 

[java] view plain copy
 
  1. package com.xjj.anes.schedule;  
  2.   
  3. public class StartupTask implements Runnable{  
  4.     @Override  
  5.     public void run() {  
  6.         //doSomething();  
  7.     }  
  8. }  

 

 


第二步:Spring配置

 

 

[java] view plain copy
 
  1. <bean id="startupTask" class="com.xjj.anes.schedule.StartupTask" />  
  2. <bean id="startupScheduledTask" class="org.springframework.scheduling.concurrent.ScheduledExecutorTask">  
  3.     <property name="delay" value="3000" />  
  4.     <property name="period" value="0" /> <!-- 0表示只执行一次,毫秒 -->  
  5.     <property name="runnable" ref="startupTask" />  
  6. </bean>  
  7.   
  8. <bean id="timerFactory" class="org.springframework.scheduling.concurrent.ScheduledExecutorFactoryBean">  
  9.     <property name="scheduledExecutorTasks">  
  10.         <list>  
  11.             <ref bean="startupScheduledTask" />  
  12.         </list>  
  13.     </property>  
  14. </bean>  

 

或直接使用注解:

 

[html] view plain copy
 
  1. <task:annotation-driven scheduler="myScheduler" />  
  2. <task:scheduler id="myScheduler" pool-size="10" />  

 

[java] view plain copy
 
  1. @Scheduled(fixedDelay=5000)    
  2. public void doSomething() {    
  3.     System.out.println("每5秒钟执行一次。");    
  4. }    
  5.   
  6. @Scheduled(fixedRate=5000)  
  7. public void doSomething() {  
  8.     // something that should execute periodically  
  9. }  
  10.   
  11. @Scheduled(initialDelay=1000, fixedRate=5000)  
  12. public void doSomething() {  
  13.     // something that should execute periodically  
  14. }  
  15.   
  16. @Scheduled(cron="*/5 * * * * MON-FRI")  
  17. public void doSomething() {  
  18.     // something that should execute on weekdays only  
  19. }  

 

或完全使用配置:

 

[html] view plain copy
 
  1. <task:scheduled-tasks scheduler="myScheduler">  
  2.     <task:scheduled ref="beanA" method="methodA" fixed-delay="5000" initial-delay="1000"/>  
  3.     <task:scheduled ref="beanB" method="methodB" fixed-rate="5000"/>  
  4.     <task:scheduled ref="beanC" method="methodC" cron="*/5 * * * * MON-FRI"/>  
  5. </task:scheduled-tasks>  
  6.   
  7. <task:scheduler id="myScheduler" pool-size="10"/>  


第三种:Spring与Quartz整合实现定时任务调度

 

 

第一步:引入相关的依赖包

[java] view plain copy
 
  1. <dependency>  
  2.     <groupId>org.springframework</groupId>  
  3.     <artifactId>spring-context-support</artifactId>  
  4.     <version>4.1.2.RELEASE</version>  
  5. </dependency>  
  6. <dependency>  
  7.     <groupId>org.quartz-scheduler</groupId>  
  8.     <artifactId>quartz</artifactId>  
  9.     <version>2.2.1</version>  
  10. </dependency>  
第二步:建立一个Java类,不需要继承如何类或实现任何接口

 

 

[java] view plain copy
 
  1. public class MyTask{  
  2. public void putDataToCache() {  
  3.         log.info("Alert! putDataToCache........" + new Date());  
  4.     }  
  5. }  

第三步:Spring配置

 

[html] view plain copy
 
  1. <bean id="myTask" class="com.xjj.anes.schedule.MyTask" />  
  2.   
  3. <!-- 调度触发器 -->  
  4. <bean id="putDataToCacheTrigger" class="org.springframework.scheduling.quartz.CronTriggerFactoryBean">  
  5.     <property name="jobDetail">  
  6.         <!-- 设置被调度的对象和方法 -->  
  7.         <bean class="org.springframework.scheduling.quartz.MethodInvokingJobDetailFactoryBean">  
  8.             <property name="concurrent" value="false"/>  
  9.             <property name="targetObject" ref="myTask"/>  
  10.             <property name="targetMethod" value="putDataToCache"/>  
  11.         </bean>  
  12.     </property>  
  13.     <property name="cronExpression" value="0 0 0/2 * * ?"/> <!-- 每两个小时执行一次 -->  
  14. </bean>  
  15.   
  16. <!-- 调度工厂 -->  
  17. <bean class="org.springframework.scheduling.quartz.SchedulerFactoryBean">  
  18.     <property name="triggers">  
  19.     <list>  
  20.         <ref bean="putDataToCacheTrigger" />   
  21.     </list>  
  22.     </property>  
  23.     <property name="startupDelay" value="6"/>  
  24. </bean>  

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

0 6 * * * 每天早上6点 
0 */2 * * * 每两个小时 
0 23-7/2,8 * * * 晚上11点到早上8点之间每两个小时,早上八点 
0 11 4 * 1-3 每个月的4号和每个礼拜的礼拜一到礼拜三的早上11点 
0 4 1 1 *     1月1日早上4点 

特殊字符 意义
*   表示所有值;
?   表示未说明的值,即不关心它为何值;'?'也代表 '任何值', 但只用于“日期”和“星期几”,当其中一个设置了条件时,另外一个就要用'?' 来表示 '任何值';
-   表示一个指定的范围;
,   表示附加一个可能值;
/   符号前表示开始时间,符号后表示每次递增的值;
L("last")   ("last") "L" 用在day-of-month字段意思是 "这个月最后一天";用在 day-of-week字段, 它简单意思是 "7" or "SAT"。如果在day-of-week字段里和数字联合使用,它的意思就是 "这个月的最后一个星期几" – 例如: "6L" means "这个月的最后一个星期五". 当我们用“L”时,不指明一个列表值或者范围是很重要的,不然的话,我们会得到一些意想不到的结果。
W("weekday")   只能用在day-of-month字段。用来描叙最接近指定 天的工作日(周一到周五)。例如:在day-of-month字段用“15W”指“最接近这个月第15天的工作日”,即如果这个月第15天是周六,那么触 发器将会在这个月第14天即周五触发;如果这个月第15天是周日,那么触发器将会在这个月第16 天即周一触发;如果这个月第15天是周二,那么就在触发器这天触发。注意一点:这个用法只会在当前月计算值,不会越过当前月。“W”字符仅能在day- of-month指明一天,不能是一个范围或列表。也可以用“LW”来指定这个月的最后一个工作日。
#   只能用在day-of-week字段。用来指定这个月的第几 个周几。例:在day-of-week字段用"6#3"指这个月第3个周五(6指周五,3指第3个)。如果指定的日期不存在,触发器就不会触发。
C   指和calendar联系后计算过的值。例:在day- of-month 字段用“5C”指在这个月第5天或之后包括calendar的第一天;在day-of-week字段用“1C”指在这周日或之后包括calendar的第 一天。


(原创文章,转载请注明转自Clement-Xu的csdn博客:http://blog.csdn.net/clementad/article/details/42042111

 

posted on 2018-05-16 16:46  J'KYO  阅读(806)  评论(0编辑  收藏  举报