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的第一天。 |