Spring实现定时启动任务
1、 Spring的定时启动任务的流程
<1>、在配置信息中SchedulerFactoryBean定义了多个触发器,在启动Spring后,就是从这个配置获取到触发器。
<2>、触发器是通过CronTriggerBean来配置,它设置了任务类和任务启动的时间
<3>、MethodInvokingJobDetailFactoryBean是任务类的配置,设置了执行任务的对象是那个Bean和需要执行这个Bean大那个方法。
2、 具体实现的例子
<1>、创建一个类HelloWord.java和它的方法sayHello().
package com.test;
public class HelloWorld {
public void sayHello() {
System.out.println("Hello world!");
}
}
<2>、在Spring的配置文件中添加HelloWord.java的配置
<bean name="hello" class="com.test.HelloWorld"></bean>
<3>、在Spring的配置文件中使用MethodInvokingJobDetailFactoryBean添加任务类的配置,在这个配置中设置了targetObject和targetMethod两个属性,即要执行的任务对象和任务方法。
<bean id="helloJobDetail" class="org.springframework.scheduling.quartz.MethodInvokingJobDetailFactoryBean">
<property name="targetObject">
<ref bean="taskJob" />
</property>
<property name="targetMethod">
<value>SayHello</value>
</property>
</bean>
<4>、使用org.springframework.scheduling.quartz.CronTriggerBean在Spring的配置文件添加触发器的配置,设置的jobDetail和cronExpression分别对应任务类和启动时间.
<!-- 配置触发器 -->
<bean id="helloTrigger" class="org.springframework.scheduling.quartz.CronTriggerBean">
<!-- 这里不可以直接在属性jobDetail中引用taskJob,因为他要求的是一个jobDetail类型的对象,所以我们得通过MethodInvokingJobDetailFactoryBean来转一下 -->
<property name="jobDetail">
<ref bean="helloJobDetail" />
</property>
<!-- 每天的8点到21点每隔1分钟触发-->
<property name="cronExpression">
<value>0 * 08-21 * * ?</value>
</property>
</bean>
<5>、使用org.springframework.scheduling.quartz.SchedulerFactoryBean把触发器放在一个集合中。
<bean class="org.springframework.scheduling.quartz.SchedulerFactoryBean">
<!-- 添加触发器 -->
<property name="triggers">
<list>
<ref local="helloTrigger" />
</list>
</property>
</bean>
<6>、创建测试类
package com.test;
import org.springframework.context.support.ClassPathXmlApplicationContext;
public class TestSpring {
public static void main(String[] args) {
new ClassPathXmlApplicationContext("spring.xml");
}
}
3、 在配置触发器是使用cronExpression的属性来配置启动时间
如:
<!-- 每天的8点到21点每隔1分钟触发-->
<property name="cronExpression">
<value>0 * 08-21 * * ?</value>
</property>
Valu的值是使用空格分开的,分别是:
秒 分钟 小时日期 月份 星期 年(可选)
字段 |
允许值 |
允许的特殊字符 |
秒 |
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 *
特殊字符 |
|
意义 |
* |
|
表示所有值; |
? |
|
表示未说明的值,即不关心它为何值; |
- |
|
表示一个指定的范围; |
, |
|
表示附加一个可能值; |
/ |
|
符号前表示开始时间,符号后表示每次递增的值; |
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的第一天。 |