概述
各种企业应用几乎都会碰到任务调度的需求,就拿论坛来说:每隔半个小时生成精华文章的RSS文件,每天凌晨统计论坛用户的积分排名,每隔30分钟执行锁定 用户解锁任务。对于一个典型的MIS系统来说,在每月1号凌晨统计上个月各部门的业务数据生成月报表,每半个小时查询用户是否已经有快到期的待处理业 务……,这样的例子俯拾皆是,不胜枚举。
Quartz 在开源任务调度框架中的翘首,它提供了强大任务调度机制,难能可贵的是它同时保持了使用的简单性。Quartz 允许开发人员灵活地定义触发器的调度时间表,并可以对触发器和任务进行关联映射。此外,Quartz提供了调度运行环境的持久化机制,可以保存并恢复调度 现场,即使系统因故障关闭,任务调度现场数据并不会丢失。此外,Quartz还提供了组件式的侦听器、各种插件、线程池等功能。
Spring为创建Quartz的Scheduler、Trigger和JobDetail提供了便利的FactoryBean类,以便能够在 Spring 容器中享受注入的好处。此外Spring还提供了一些便利工具类直接将Spring中的Bean包装成合法的任务。Spring进一步降低了使用 Quartz的难度,能以更具Spring风格的方式使用Quartz。概括来说它提供了两方面的支持:
1)为Quartz的重要组件类提供更具Bean风格的扩展类;
2)提供创建Scheduler的BeanFactory类,方便在Spring环境下创建对应的组件对象,并结合Spring容器生命周期进行启动和停止的动作。
创建JobDetail
你可以直接使用Quartz的JobDetail在Spring中配置一个JobDetail Bean,但是JobDetail使用带参的构造函数,对于习惯通过属性配置的Spring用户来说存在使用上的不便。为此Spring通过扩展 JobDetail提供了一个更具Bean风格的JobDetailBean。此外,Spring提供了一个 MethodInvokingJobDetailFactoryBean,通过这个FactoryBean可以将Spring容器中Bean的方法包装成 Quartz任务,这样开发者就不必为Job创建对应的类。
JobDetailBean
JobDetailBean扩展于Quartz的JobDetail。使用该Bean声明JobDetail时,Bean的名字即是任务的名字,如果没有指定所属组,即使用默认组。除了JobDetail中的属性外,还定义了以下属性:
● jobClass:类型为Class,实现Job接口的任务类;
● beanName:默认为Bean的id名,通过该属性显式指定Bean名称,它对应任务的名称;
● jobDataAsMap:类型为Map,为任务所对应的JobDataMap提供值。之所以需要提供这个属性,是因为除非你手工注册一 个编辑器,你不能直接配置JobDataMap类型的值,所以Spring通过jobDataAsMap设置JobDataMap的值;
●applicationContextJobDataKey:你可以将Spring ApplicationContext的引用保存到JobDataMap中,以便在Job的代码中访 问ApplicationContext。为了达到这个目的,你需要指定一个键,用以在jobDataAsMap中保存 ApplicationContext,如果不设置此键,JobDetailBean就不将ApplicationContext放入到 JobDataMap中;
●jobListenerNames:类型为String[],指定注册在Scheduler中的JobListeners名称,以便让这些监听器对本任务的事件进行监听。
<!-- Code highlighting produced by Actipro CodeHighlighter (freeware) http://www.CodeHighlighter.com/ --> < bean name = " jobDetail " class = " org.springframework.scheduling.quartz.JobDetailBean " > < property name = " jobClass " value = " com.baobaotao.quartz.MyJob " /> < property name = " jobDataAsMap " > ①
< map > < entry key = " size " value = " 10 " /> </ map > </ property > < property name = " applicationContextJobDataKey " value = " applicationContext " /> ②
</ bean >
<!-- Code highlighting produced by Actipro CodeHighlighter (freeware) http://www.CodeHighlighter.com/ --> package com.baobaotao.quartz;
…
import org.quartz.Job;
import org.quartz.JobExecutionContext;
import org.quartz.JobExecutionException;
import org.springframework.context.ApplicationContext;
public class MyJob implements Job {
public void execute(JobExecutionContext jctx) throws JobExecutionException {
Map dataMap = jctx.getJobDetail().getJobDataMap();①获取JobDetail关联的JobDataMap
String size = (String)dataMap. get ( " size " );②
ApplicationContext ctx = (ApplicationContext)dataMap. get ( " applicationContext " );③
System. out .println( " size: " + size);
dataMap.put( " size " ,size + " 0 " );④对JobDataMap所做的更改是否被会持久,取决于任务的类型
// do sth... } }
<!-- Code highlighting produced by Actipro CodeHighlighter (freeware) http://www.CodeHighlighter.com/ --> < bean id = " jobDetail_1 " class = " org.springframework.scheduling.quartz.MethodInvokingJobDetailFactoryBean " > < property name = " targetObject " ref = " myService " /> ① 引用一个Bean
< property name = " targetMethod " value = " doJob " /> ② 指定目标Bean的方法
< property name = " concurrent " value = " false " /> ③ 指定最终封装出的任务是否有状态
< bean id = " myService " class = " com.baobaotao.service.MyService " />
<!-- Code highlighting produced by Actipro CodeHighlighter (freeware) http://www.CodeHighlighter.com/ --> package com.baobaotao.service;
public class MyService {
public void doJob() {①被封装成任务的目标方法
System. out .println( " in MyService.dojob(). " );
} }
<!-- Code highlighting produced by Actipro CodeHighlighter (freeware) http://www.CodeHighlighter.com/ --> < bean id = " simpleTrigger " class = " org.springframework.scheduling.quartz.SimpleTriggerBean " > < property name = " jobDetail " ref = " jobDetail " /> < property name = " startDelay " value = " 1000 " /> < property name = " repeatInterval " value = " 2000 " /> < property name = " repeatCount " value = " 100 " /> < property name = " jobDataAsMap " > ①
< map > < entry key = " count " value = " 10 " /> </ map > </ property > </ bean >
<!-- Code highlighting produced by Actipro CodeHighlighter (freeware) http://www.CodeHighlighter.com/ --> package com.baobaotao.quartz;
…
public class MyJob implements StatefulJob {
public void execute(JobExecutionContext jctx) throws JobExecutionException {
Map dataMap = jctx.getTrigger().getJobDataMap();①获取Trigger的JobDataMap
String count = dataMap. get ( " count " );
dataMap.put(“count”,” 30 ”) ② 对JobDataMap的更改不会被持久,不影响下次的执行
…
} }
<!-- Code highlighting produced by Actipro CodeHighlighter (freeware) http://www.CodeHighlighter.com/ --> < bean id = " checkImagesTrigger " class = " org.springframework.scheduling.quartz.CronTriggerBean " > < property name = " jobDetail " ref = " jobDetail " /> < property name = " cronExpression " value = " 0/5 * * * * ? " /> </ bean >
<!-- Code highlighting produced by Actipro CodeHighlighter (freeware) http://www.CodeHighlighter.com/ --> < bean id = " scheduler " class = " org.springframework.scheduling.quartz.SchedulerFactoryBean " > < property name = " triggers " > ①注册多个Trigger
< list > < ref bean = " simpleTrigger " /> </ list > </ property > < property name = " schedulerContextAsMap " > ②以Map类型设置SchedulerContext数据
< map > < entry key = " timeout " value = " 30 " /> </ map > </ property >
<!-- Code highlighting produced by Actipro CodeHighlighter (freeware) http://www.CodeHighlighter.com/ --> < bean id = " scheduler " class = " org.springframework.scheduling.quartz.SchedulerFactoryBean " > …
< property name = " quartzProperties " > < props > < prop key = " org.quartz.threadPool.class " > ①Quartz属性项1
org.quartz.simpl.SimpleThreadPool
</ prop > < prop key = " org.quartz.threadPool.threadCount " > 10 </ prop > ①Quartz属性项2
</ props > </ property > </ bean >