配置quartz 在spring中需要三个jar包:
quartz-1.8.5.jar、commons-collections-3.2.1.jar、commons-logging-1.1.jar
首先要配置我们的spring.xml
xmlns 多加下面的内容、
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.1.xsd
最后是我们的task任务扫描注解
<task:annotation-driven/>
我的配置扫描位置是:
<bean class=”org.springframework.beans.factory.annotation.AutowiredAnnotationBeanPostProcessor”/>
<context:component-scan base-package=”com.test”/>
扫描的是com.test这样的包下的内容、
下面需要接口和实现(我的这几个java文件都是com.test的包下的、)
1
2
3
|
public
interface
IMyTestService {
public
void
myTest();
}
|
1
2
3
4
5
6
7
8
|
@Component
//import org.springframework.stereotype.Component;
public
class
MyTestServiceImpl
implements
IMyTestService {
@Scheduled
(cron=
"0/5 * * * * ? "
)
//每5秒执行一次
@Override
public
void
myTest(){
System.out.println(
"进入测试"
);
}
}
|
执行后控制台就会打印出 进入测试 了
需要注意的几点:
1、spring的@Scheduled注解 需要写在实现上、
2、 定时器的任务方法不能有返回值(如果有返回值,spring初始化的时候会告诉你有个错误、需要设定一个proxytargetclass的某个值为true、具体就去百度google吧)
3、实现类上要有组件的注解@Component
剩下的就是corn表达式了、具体使用以及参数请百度google、
【秒】 【分】 【时】 【日】 【月】 【周】 【年】
下面只例出几个式子
CRON表达式 含义
“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触发
有时候我们的任务(Job)需要再某些任务完成之后才能进行;例如从旧的数据库批量导数据的时候;需要现将被其他数据依赖的数据导入新的数据库;然后再进行关系的导入.。在这种情况下我们就可以使用Quartz的listener来做文章了。
首先我们写一个主任务的类,命名为MainJob;她的作用是作为一系列任务的开始点。
MainJob.java
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
|
package
jobs;
import
org.apache.log4j.Logger;
import
org.quartz.JobExecutionContext;
import
org.quartz.JobExecutionException;
import
org.springframework.scheduling.quartz.QuartzJobBean;
public
class
MainJob
extends
QuartzJobBean {
private
Logger logger=Logger.getLogger(getClass());
@Override
protected
void
executeInternal(JobExecutionContext arg0)
throws
JobExecutionException {
// TODO Auto-generated method stub
logger.debug(
"Just say hi."
);
}
}
|
然后我们新建另外一个任务(SecondJob)作为后续任务:
SecondJob.java
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
|
package
jobs;
import
org.apache.log4j.Logger;
import
org.quartz.JobExecutionContext;
import
org.quartz.JobExecutionException;
import
org.springframework.scheduling.quartz.QuartzJobBean;
public
class
SecondJob
extends
QuartzJobBean {
private
Logger logger=Logger.getLogger(getClass());
@Override
protected
void
executeInternal(JobExecutionContext arg0)
throws
JobExecutionException {
// TODO Auto-generated method stub
logger.debug(
"I'm the second job."
);
}
}
|
创建一个TriggerListener,重写其triggerComplete方法,并且添加一些方便spring注入的属性和方法。
NextJobTriggerListener.java
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
|
package
listeners;
import
org.apache.log4j.Logger;
import
org.quartz.JobDetail;
import
org.quartz.JobExecutionContext;
import
org.quartz.Scheduler;
import
org.quartz.Trigger;
import
org.quartz.listeners.TriggerListenerSupport;
import
org.springframework.scheduling.quartz.QuartzJobBean;
import
org.springframework.scheduling.quartz.SimpleTriggerBean;
public
class
NextJobTriggerListener
extends
TriggerListenerSupport {
private
Logger logger=Logger.getLogger(getClass());
private
String name;
public
String getName() {
return
this
.name;
}
public
void
setName(String name)
{
this
.name=name;
}
private
SimpleTriggerBean nextTrigger;
public
void
setNextTrigger(SimpleTriggerBean nextTrigger) {
this
.nextTrigger=nextTrigger;
}
@Override
public
void
triggerComplete(Trigger trigger, JobExecutionContext context,
int
code) {
try
{
Scheduler schduler=context.getScheduler();
JobDetail nextJob=nextTrigger.getJobDetail();
//查找名称和即将加入的任务一样的任务
JobDetail oldJob=schduler.getJobDetail(nextJob.getName(),nextJob.getGroup());
//查找名称和即将加入的触发器一样的触发器
Trigger oldTrigger=schduler.getTrigger(nextTrigger.getName(),nextTrigger.getGroup());
if
(oldJob==
null
&&oldTrigger==
null
)
//同名的任务和触发器都不存在
{
logger.debug(
"inside scheduleJob."
+code);
schduler.scheduleJob(nextJob,nextTrigger);
}
else
//同名的任务或触发器
{
logger.debug(
"oldJob==null:"
+(oldJob==
null
));
logger.debug(
"oldTrigger==null:"
+(oldTrigger==
null
));
}
super
.triggerComplete(trigger, context, code);
}
catch
(Exception e)
{
e.printStackTrace();
}
}
}
|
配置spring 的applicationContext.xml
applicationContext.xml
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
|
<?
xml
version
=
"1.0"
encoding
=
"UTF-8"
?>
xsi:schemaLocation="http://www.springframework.org/schema/beanshttp://www.springframework.org/schema/beans/spring-beans-2.0.xsd
http://www.springframework.org/schema/jee http://www.springframework.org/schema/jee/spring-jee-2.0.xsd">
<!-- 主任务 -->
<
bean
id
=
"mainJob"
class
=
"org.springframework.scheduling.quartz.JobDetailBean"
>
<!-- 运行的类 -->
<
property
name
=
"jobClass"
>
<
value
> jobs.MainJob </
value
>
</
property
>
</
bean
>
<!-- 主任务的监听器 -->
<
bean
id
=
"mainTriggerListener"
class
=
"listeners.NextJobTriggerListener"
>
<!-- 下个触发器 -->
<
property
name
=
"nextTrigger"
ref
=
"secondTrigger"
></
property
>
<!-- 监听器名称 -->
<
property
name
=
"name"
value
=
"mainTriggerListener"
></
property
>
</
bean
>
<!-- 主任务的触发器 -->
<
bean
id
=
"mainTrigger"
class
=
"org.springframework.scheduling.quartz.SimpleTriggerBean"
>
<
property
name
=
"jobDetail"
>
<!-- 上面创建的任务调度对象 -->
<
ref
bean
=
"mainJob"
/>
</
property
>
<!-- 启动60秒后执行任务调度的excute方法 -->
<
property
name
=
"startDelay"
>
<
value
> 6000 </
value
>
</
property
>
<!-- 运行次数 -->
<
property
name
=
"repeatCount"
>
<
value
> </
value
>
</
property
>
<!-- 隔一个小时运行一次(貌似多余,不写会报错) -->
<
property
name
=
"repeatInterval"
>
<
value
> 3600000 </
value
>
</
property
>
<
property
name
=
"triggerListenerNames"
>
<
list
>
<
value
> mainTriggerListener </
value
>
</
list
>
</
property
>
</
bean
>
<!-- 后续任务 -->
<
bean
id
=
"secondJob"
class
=
"org.springframework.scheduling.quartz.JobDetailBean"
>
<!-- 运行的类 -->
<
property
name
=
"jobClass"
>
<
value
> jobs.SecondJob </
value
>
</
property
>
</
bean
>
<!-- 后续任务的触发器 -->
<
bean
id
=
"secondTrigger"
class
=
"org.springframework.scheduling.quartz.SimpleTriggerBean"
>
<
property
name
=
"jobDetail"
>
<!-- 上面创建的任务调度对象 -->
<
ref
bean
=
"secondJob"
/>
</
property
>
<!-- 启动6秒后执行任务调度的excute方法 -->
<
property
name
=
"startDelay"
>
<
value
> 6000 </
value
>
</
property
>
<!-- 运行次数 -->
<
property
name
=
"repeatCount"
>
<
value
> </
value
>
</
property
>
<!-- 隔一个小时运行一次(貌似多余,不写会报错) -->
<
property
name
=
"repeatInterval"
>
<!--
<value>3600000</value>
-->
<
value
> 6000 </
value
>
</
property
>
</
bean
>
<!-- 任务调度工厂类 -->
<
bean
class
=
"org.springframework.scheduling.quartz.SchedulerFactoryBean"
>
<!-- 这一部分的配置不用管 -->
<
property
name
=
"quartzProperties"
>
<
props
>
<
prop
key
=
"org.quartz.threadPool.class"
>
org.quartz.simpl.SimpleThreadPool
</
prop
>
<
prop
key
=
"org.quartz.threadPool.threadCount"
> </
prop
>
<
prop
key
=
"org.quartz.threadPool.threadPriority"
>
</
prop
>
<
prop
key
=
"org.quartz.threadPool.threadsInheritContextClassLoaderOfInitializingThread"
>
true
</
prop
>
</
props
>
</
property
>
<!-- 触发器,可以放一大堆触发器 -->
<
property
name
=
"triggers"
>
<
list
>
<!-- 在这里加 -->
<
ref
bean
=
"mainTrigger"
/>
</
list
>
</
property
>
<
property
name
=
"triggerListeners"
>
<
list
>
<!-- 触发器的监听器 -->
<
ref
bean
=
"mainTriggerListener"
/>
</
list
>
</
property
>
</
bean
>
</
beans
>
|
开启服务器,输出
1
2
3
4
5
|
DEBUG [ MainJob.executeInternal(14) ] Just say hi.
DEBUG [ NextJobTriggerListener.triggerComplete(38) ] inside scheduleJob .3
DEBUG [SecondJob.executeInternal(14)] I'm the second job.
DEBUG [ NextJobTriggerListener.triggerComplete(43) ] oldJob==null:
false
DEBUG [ NextJobTriggerListener.triggerComplete(44) ] oldTrigger== null:
false
|