· 在Spring中使用Quartz的方法就不介绍了,个人感觉,Quartz功能复杂,这也导致了其过于笨重,没有特殊的复杂需求,尽可以使用Spring的scheduled-tasks
· Spring定时任务scheduled-tasks
Spring的scheduled-tasks,可以通过注解及配置文件两种方式实现,这里仅介绍配置文件的使用方法(本处不对配置文件及注解谁优谁劣做争辩)
JAR
至少需要引入spring-core,建议使用MAVEN管理,可同时自动引入依赖包,spring-core的dependency可由此处查询http://mvnrepository.com/artifact/org.springframework/spring-core
XMLNS&XSI
使用scheduled-tasks,必须引入xmlns:task="http://www.springframework.org/schema/task",同时必须在xsi:schemaLocation中引入http://www.springframework.org/schema/task及http://www.springframework.org/schema/task/spring-task-3.2.xsd,注意,spring-task的xsd版本依引入的spring-core版本不同而不同。
使用方法
xml中可按照如下配置
<task:scheduled-tasks>
<!-- 每小时 -->
<task:scheduled ref="Timer" method="runCron" cron="0 0 */1 * * ?" />
<!-- 每分钟 -->
<task:scheduled ref="Timer" method="runFixed" fixed-rate="60000" initial-delay="30000" />
</task:scheduled-tasks>
method 需要执行的方法
cron CRON定时表达式
fixed-rate 指定多久执行一次,单位,毫秒
initial-delay 与fixed-rate配合使用,指定srping初始化完成之后,延迟多久开始执行,单位,毫秒
initial-delay 与fixed-rate配合使用,指定srping初始化完成之后,延迟多久开始执行,单位,毫秒
上述配置中,第一个定时器的意思为,每小时的0分0秒,执行CronTimer类中的runCron方法
第二个定时器的意思为,spring初始化后30秒开始,每隔60秒执行一次FixedTimer类中的runFixed方法
对于cron表达式,其字段(以空格分隔)分别为:
秒(0~59)
分钟(0~59)
小时(0~23)
月份中的日期(1~31,但是你需要考虑你月的天数)
月份(1~12或者JAN~DEC)
星期中的日期(1~7 或 SUN~SAT)
年份(1970~2099)
各字段可取值
秒0-59 , - * /
分0-59 , - * /
时0-23 , - * /
日1-31 , - * ? / L W C
日1-31 , - * ? / L W C
月1-12 or JAN-DEC , - * /
周1-7 or SUN-SAT , - * ? / L C #
周1-7 or SUN-SAT , - * ? / L C #
年(可选字段) empty, 1970-2099 , - * /
可用值详细分析如下:
“*”——字符可以用于所有字段。比如在“分”字段中设为"*"表示"每一分钟"的含义。
“?”——字符可以用在“日”和“周”字段,它用来指定'不明确的值',这在你需要指定这两个字段中的某一个值而不是另外一个的时候会被用到。在后面的例子中可以看到其含义。
“-”——字符被用来指定一个值的范围。比如在“小时”字段中设为"10-12"表示"10点到12点"。
“,”——字符指定数个值。比如在“周”字段中设为"MON,WED,FRI"表示"周一、周三和周五"。
“/”——字符用来指定一个值的的增加幅度。比如在“秒”字段中设置为"0/15"表示"第0, 15, 30,和45秒",而"5/15"则表示"第5, 20, 35,和50",在'/'前加"*"字符相当于指定从0秒开始。每个字段都有一系列可以开始或结束的数值。对于“秒”和“分”字段来说,其数值范围为0到59,对于“小时”字段来说其为0到23,对于“日”字段来说为0到31,而对于“月”字段来说为1到12。"/"字段前表示从哪个值开始,"/"字段后表示增加的幅度。
“L”——字符可用在“日”和“周”这两个字段,它是"last"的缩写,但是在这两个字段中有不同的含义。例如,“日”字段中的"L"表示"一个月中的最后一天" ,对于1月就是31号,对于2月来说就是28号(非闰年)。而在“周几”字段中,它简单的表示"7"或"SAT",但是如果在“周几”字段中使用时跟在某个数字之后,它表示"该月最后一个星期×",比如"6L"表示"该月最后一个周五"。当使用'L'选项时,指定确定的列表或者范围非常重要,否则你会被结果搞糊涂的。
“W”——可用于“日”字段。用来指定离给定日期最近的工作日(周一到周五)。比如你将“日”字段设为"15W",意为: "离该月15号最近的工作日"。因此如果15号为周六,触发器会在14号即周五调用,如果15号为周日,触发器会在16号也就是周一触发,如果15号为周二,那么当天就会触发。然而如果你将“日”字段设为"1W",而一号又是周六,触发器会于下周一也就是当月的3号触发,因为它不会越过当月的值的范围边界。'W'字符只能用于“日”字段的值为单独的一天而不是一系列值的时候。
“L”和“W”可以组合用于“日”字段表示为'LW',意为"该月最后一个工作日"。
“#”——字符可用于“周几”字段。该字符表示“该月第几个周×”,比如"6#3"表示该月第三个周五( 6表示周五而"#3"该月第三个)。再比如"2#1"表示该月第一个周一,而"4#5"表示该月第五个周三。注意,如果你指定"#5"该月没有第五个“周×”,该月是不会触发的。
“C”——字符可用于“日”和“周几”字段,它是"calendar"的缩写,它表示为基于相关的日历所计算出的值(如果有的话)。如果没有关联的日历,那它等同于包含全部日历。“日”字段值为"5C"表示"日历中的第一天或者5号以后",“周几”字段值为"1C"则表示"日历中的第一天或者周日以后"。
对于“月份”字段和“周几”字段来说合法的字符都不是大小写敏感的。
以下给出一些示例
"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触发
"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触发
"0 15 10 15 * ?" 每月15号的10:15触发
"0 15 10 L * ?" 每月的最后一天的10:15触发
"0 15 10 ? * 6L" 每月最后一个周五的10:15
对于bean的配置,不在赘述,如下
<bean id="Timer" class="com.manerfan.timer.Timer" />
package com.manerfan.timer;
import java.util.Date;
public class Timer {
public void cronTimer() {
System.out.println("For CronTimer [" + new Date() + "]");
}
public void fixedTimer() {
System.out.println("For FixedTimer [" + new Date() + "]");
}
}
对于定时任务类中的定时方法,建议在线程中运行,一种方法可以自己写线程,另一种方法可以直接使用Spring的@Async注解
使用@Async注解需引入spring-context及spring-context-support两个jar,xmlns:context="http://www.springframework.org/schema/context" ,同时需在xsi:schemaLocation中引入http://www.springframework.org/schema/context及http://www.springframework.org/schema/context/spring-context-3.2.xsd,同样的,spring-context的xsd版本依引入的spring-core版本不同而不同。
使用注解时,需添加
<context:component-scan base-package="com.manerfan.timer"></context:component-scan>
增加对注解的扫描
同时添加
<task:executor id="executor" pool-size="10" />
<task:annotation-driven executor="executor" />
开启 @Scheduled @Async 等注解
此时修改代码
package com.manerfan.timer;
import java.util.Date;
import org.springframework.scheduling.annotation.Async;
public class Timer {
@Async
public void cronTimer() {
System.out.println("For CronTimer [" + new Date() + "]");
}
@Async
public void fixedTimer() {
System.out.println("For FixedTimer [" + new Date() + "]");
}
}
Spring便会自动将cronTimer及fixedTimer方法放入线程中运行