`
zl198751
  • 浏览: 279485 次
  • 性别: Icon_minigender_1
  • 来自: 上海
社区版块
存档分类
最新评论

Spring的事件调度Quartz

阅读更多

spring包装了Quartz,是我们可以基于xml配置实现quartz。

首先说明必要的3个角色:

Job:是执行的业务逻辑,可以把类封装成job或则仅仅封装一个类中方法。Trigger:是调度的触发器,暂时暂时只能是时间。 Scheduler:调度器,将trigger都放入其中。

Job需要使用其他的service方法则需要将applicationContex注入类中,再得到service类或者将将类直接注入Job 中。

具体的配置(载录网络内容http://tech.it168.com/oldarticle/2007-04-25 /200704250927125.shtml):

    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名称,以便让这些监听器对本任 务的事件进行监听。

    下面配置片断使用JobDetailBean在Spring中配置一个JobDetail:

<!-- {12838466023360}-->< 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 >

    JobDetailBean封装了MyJob任务类,并为Job对应JobDataMap设置了一个size的数据。此外,通过指定 applicationContextJobDataKey让Job的JobDataMap持有Spring ApplicationContext的引用。

 

这样,MyJob在运行时就可以通过JobDataMap访问到size和 ApplicationContext了。来看一下MyJob的代码,如代码清单 8所示:

 

    代码清单 8 MyJob

 

<!-- {12838466023361}-->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... } }

    在②处获取size值,在③处还可以根据键“applicationContext”获取ApplicationContext,有了 ApplicationContext的引用,Job就可以毫无障碍访问Spring容器中的任何Bean了。MyJob可以在execute()方法中 对JobDataMap进行更改,如④所示。如果MyJob实现Job接口,这个更改对于下一次执行是不可见的,如果MyJob实现 StatefulJob接口,这种更改对下一次执行是可见的。

MethodInvokingJobDetailFactoryBean
    通常情况下,任务都定义在一个业务类方法中。这时,为了满足Quartz Job接口的规定,还需要定义一个引用业务类方法的实现类。为了避免创建这个只包含一行调用代码的Job实现类,Spring为我们提供了 MethodInvokingJobDetailFactoryBean,借由该FactoryBean,我们可以将一个Bean的某个方法封装成满足 Quartz要求的Job。来看一个具体的例子:

<!-- {12838466023362}-->< 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 " />

    jobDetail_1将MyService#doJob()封装成一个任务,同时通过concurrent属性指定任务的类型,默认情况下封装为无状态 的任务,如果希望目标封装为有状态的任务,仅需要将concurrent设置为false就可以了。Spring通过名为concurrent的属性指定 任务的类型,能够更直接地描述到任务执行的方式(有状态的任务不能并发执行,无状态的任务可并发执行),对于不熟悉Quartz内部机制的用户来说,比起 statefule,concurrent显然更简明达意些。

 

MyService服务类拥有一个doJob()方法,它的代码如下所示:

 

<!-- {12838466023363}-->package com.baobaotao.service; public class MyService ... { public void doJob() ... {①被封装成任务的目标方法 System. out .println( " in MyService.dojob(). " ); } }

    doJob()方法即可以是static,也可以是非static的,但不能拥有方法入参。通过 MethodInvokingJobDetailFactoryBean产生的JobDetail不能被序列化,所以不能被持久化到数据库中的,如果希望 使用持久化任务,则你只能创建正规的Quartz的Job实现类了。

   创建Trigger
    Quartz中另一个重要的组件就是Trigger,Spring按照相似的思路分别为SimpleTrigger和CronTrigger提供了更具 Bean风格的SimpleTriggerBean和CronTriggerBean扩展类,通过这两个扩展类更容易在Spring中以Bean的方式配 置Trigger。

    SimpleTriggerBean
    默认情况下,通过SimpleTriggerBean配置的Trigger名字即为Bean的名字,并属于默认组Trigger组。 SimpleTriggerBean在SimpleTrigger的基础上,新增了以下属性:
    ● jobDetail:对应的JobDetail;
    ● beanName:默认为Bean的id名,通过该属性显式指定Bean名称,它对应Trigger的名称;
    ● jobDataAsMap:以Map类型为Trigger关联的JobDataMap提供值;
    ● startDelay:延迟多少时间开始触发,单位为毫秒,默认为0;
    ● triggerListenerNames:类型为String[],指定注册在Scheduler中的TriggerListener名称,以便让这些 监听器对本触发器的事件进行监听。
    下面的实例使用SimpleTriggerBean定义了一个Trigger,该Trigger和jobDetail相关联,延迟10秒后启动,时间间隔 为20秒,重复执行100次。此外,我们还为Trigger设置了JobDataMap数据:

<!-- {12838466023364}-->< 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 >

    需要特别注意的是,①处配置的JobDataMap是Trigger的JobDataMap,任务执行时必须通过以下方式获取配置的值:

 

<!-- {12838466023365}-->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的更改不会被持久,不影响下次的执行 } }

    CronTriggerBean

 

    CronTriggerBean扩展于CronTrigger,触发器的名字即为Bean的名字,保存在默认组中。在CronTrigger的基础上,新 增的属性和SimpleTriggerBean大致相同,配置的方法也和SimpleTriggerBean相似,下面给出一个简单的例子:

 

<!-- {12838466023366}-->< bean id = " checkImagesTrigger " class = " org.springframework.scheduling.quartz.CronTriggerBean " > < property name = " jobDetail " ref = " jobDetail " /> < property name = " cronExpression " value = " 0/5 * * * * ? " /> </ bean >

   创建Scheduler  
    Quartz的SchedulerFactory是标准的工厂类,不太适合在Spring环境下使用。此外,为了保证Scheduler能够感知 Spring容器的生命周期,完成自动启动和关闭的操作,必须让Scheduler和Spring容器的生命周期相关联。以便在Spring容器启动 后,Scheduler自动开始工作,而在Spring容器关闭前,自动关闭Scheduler。为此,Spring提供 SchedulerFactoryBean,这个FactoryBean大致拥有以下的功能:
    1)以更具Bean风格的方式为Scheduler提供配置信息;
    2)让Scheduler和Spring容器的生命周期建立关联,相生相息;
    3)通过属性配置部分或全部代替Quartz自身的配置文件。
    来看一个SchedulerFactoryBean配置的例子:
    代码清单 9 SchedulerFactoryBean配置

<!-- {12838466023367}-->< 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 >

    ③显式指定Quartz的配置文件地址
   <property name="configLocation" value="classpath:com/baobaotao/quartz/quartz.properties" />
    </bean>
    SchedulerFactoryBean的triggers属性为Trigger[]类型,可以通过该属性注册多个Trigger,在①处,我们注册了 一个Trigger。Scheduler拥有一个类似于ServletContext的SchedulerContext。 SchedulerFactoryBean允许你以Map的形式设置SchedulerContext的参数值,如②所示。默认情况下,Quartz在类 路径下查询quartz.properties配置文件,你也可以通过configLocation属性显式指定配置文件位置,如③所示。
    除了实例中所用的属性外,SchedulerFactoryBean还拥有一些常见的属性:
    ●calendars:类型为Map,通过该属性向Scheduler注册Calendar;
    ●jobDetails:类型为JobDetail[],通过该属性向Scheduler注册JobDetail;
    ●autoStartup:SchedulerFactoryBean在初始化后是否马上启动Scheduler,默认为true。如果设置为 false,需要手工启动Scheduler;
    ●startupDelay:在SchedulerFactoryBean初始化完成后,延迟多少秒启动Scheduler,默认为0,表示马上启动。如 果并非马上拥有需要执行的任务,可通过startupDelay属性让Scheduler延迟一小段时间后启动,以便让Spring能够更快初始化容器中 剩余的Bean。

   ●SchedulerFactoryBean的一个重要功能是允许你将Quartz配置文件中的信息转移到Spring配置文件中,带来的好处是,配置信 息的集中化管理,同时我们不必熟悉多种框架的配置文件结构。回忆一个Spring集成JPA、Hibernate框架,就知道这是Spring在集成第三 方框架经常采用的招数之一。SchedulerFactoryBean通过以下属性代替框架的自身配置文件:
    ●dataSource:当需要使用数据库来持久化任务调度数据时,你可以在Quartz中配置数据源,也可以直接在Spring中通过 dataSource指定一个Spring管理的数据源。如果指定了该属性,即使quartz.properties中已经定义了数据源,也会被此 dataSource覆盖;
    ●transactionManager:可以通过该属性设置一个Spring事务管理器。在设置dataSource时,Spring强烈推荐你使用一 个事务管理器,否则数据表锁定可能不能正常工作;
    ●nonTransactionalDataSource:在全局事务的情况下,如果你不希望Scheduler执行化数据操作参与到全局事务中,则可以 通过该属性指定数据源。在Spring本地事务的情况下,使用dataSource属性就足够了;
    ●quartzProperties:类型为Properties,允许你在Spring中定义Quartz的属性。其值将覆盖 quartz.properties配置文件中的设置,这些属性必须是Quartz能够识别的合法属性,在配置时,你可以需要查看Quartz的相关文 档。下面是一个配置quartzProperties属性的例子:

<!-- {12838466023368}-->< 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 >

    在实际应用中,我们并不总是在程序部署的时候就可能确定需要哪些任务,往往需要在运行期根据业务数据动态产生触发器和任务。你完全可以在运行期通过代码调 用SchedulerFactoryBean获取Scheduler实例,进行动态的任务注册和调度。



    小结

    Spring为Quartz的JobDetail和Trigger提供了更具Bean风格的支持类,这使我们能够更地方便地在Spring中通过配置定制 这些组件实例。Spring的SchedulerFactoryBean让我们可以脱离Quartz自身的配置体系,而以更具Spring风格的方式定义 Scheduler。此外,还可以享受Scheduler生命周期和Spring 容器生命周期绑定的好处。

分享到:
评论

相关推荐

    spring任务调度(Quartz )

    Spring中的任务调度是实现应用程序自动化运行任务的重要工具,而Quartz是Java领域广泛使用的开源任务调度框架。在本文中,我们将深入探讨如何在Spring中集成Quartz进行任务调度,并通过一个简单的示例来理解其工作...

    spring2与quartz在Web整合

    通过理解和掌握这些知识点,开发者能够有效地在 Spring 2 和 Quartz 之间建立桥梁,实现 Web 应用中的复杂任务调度功能。对于出现问题的整合,开发者可以通过分析源码、查看日志和调试来定位和解决问题,进一步提升...

    Spring整合任务调度框架Quartz

    Spring 整合任务调度框架 Quartz 在软件开发中,任务调度框架是非常重要的一部分,它可以帮助开发者更好地管理和执行各种任务。在 Java 领域中,Quartz 是一个非常流行的任务调度框架,而 Spring 是一个非常流行的 ...

    Spring中的Quartz配置-Spring-定时器-java定时器.doc

    在 Spring 框架中,Quartz 是一个非常流行的开源作业调度器,可以实现任务的定时执行。在本篇文章中,我们将讨论如何在 Spring 中配置 Quartz,以实现 Java 定时器的功能。 Quartz 介绍 Quartz 是一个开源的作业...

    spring3整合quartz1.8和spring3整合quartz2.2两个版本示例

    本示例将探讨如何将 Spring 3 与 Quartz 1.8 和 2.2 版本进行整合,以实现高效的任务调度。 首先,我们来看 Spring 3 整合 Quartz 1.8 的步骤: 1. **引入依赖**:在项目中添加 Quartz 和 Spring 相关的库,确保...

    Spring Quartz 调度器

    Spring Quartz 调度器是Java开发中常用的一种任务调度框架,它结合了Spring框架的强大功能与OpenSymphony的Quartz库,使得开发者能够轻松地在应用中添加定时任务。在myEclipse环境中,借助Tomcat服务器和JDK运行环境...

    spring3.0+quartz-1.6.0定时

    总结起来,"spring3.0+quartz-1.6.0定时"的实现涉及到Spring框架的定时任务支持和Quartz库的结合使用,通过Spring的配置管理Quartz调度器,定义任务和触发器,最后编写具体任务逻辑的Java类。这种组合在实际项目中...

    quartz和spring-quartz

    Quartz和Spring-Quartz是两个在Java世界中广泛使用的定时任务管理框架。Quartz是一个开源的作业调度框架,允许应用程序定义和调度任务在特定时间执行。而Spring-Quartz则是Spring框架对Quartz的集成,它使得在Spring...

    spring 任务调度

    Spring Task是Spring内置的轻量级任务调度器,而Quartz是更强大的第三方调度库,可以在Spring中集成使用。 1. **Spring Task** - **基本概念**:Spring Task是Spring框架的一部分,提供了基本的定时任务调度功能,...

    最新 Spring 4.2.2 集成 Quartz Scheduler 2.2.2 任务调度示例

    本资源是一个最新 Spring 4.2.2 集成 Quartz Scheduler 2.2.2 的一个简单的 demo,也是博客《最新 Spring 4.2.2 集成 Quartz Scheduler 2.2.2 任务调度示例》的配套示例项目,该博客地址是:...

    Spring调度器Quartz笔记

    -- 使用Spring的SchedulerFactoryBean类启动调度工作 --&gt; &lt;bean class="org.springframework.scheduling.quartz.SchedulerFactoryBean"&gt; &lt;property name="triggers"&gt; &lt;array&gt; &lt;ref bean="helloWordBeanTrigger"/&gt; ...

    spring boot + quartz 动态管理 分布式

    Spring Boot简化了Spring应用的初始搭建以及开发过程,提供了开箱即用的功能,而Quartz则是一个强大的任务调度库,常用于实现定时任务。 Spring Boot是一个基于Spring框架的快速开发工具,它通过预配置的starter ...

    spring+quartz任务调度代码版

    本项目"spring+quartz任务调度代码版"显然是一个结合了这两者的实践案例,旨在展示如何在Spring环境下集成和使用Quartz进行任务调度。 Spring框架是一个开源的应用框架,它提供了丰富的功能,包括依赖注入、AOP...

    spring+quartz demo

    【Spring + Quartz 框架整合详解】 Spring 和 Quartz 是两个在 Java 开发领域中非常重要的框架。Spring 是一个全面的、开源的应用框架,提供依赖注入(DI)、面向切面编程(AOP)以及用于简化企业级应用开发的其他...

    spring timer、quartz俩种任务调度

    总的来说,`Spring Timer`适合简单的定时任务需求,而`Quartz Scheduler`更适合处理大型项目或需要高级调度功能的场景。在实际应用中,开发者可以根据项目需求选择合适的任务调度器。通过`spring-taskScheduling`这...

    quartz整合springbatch动态集群定时实现mysql参考

    Quartz主要用于任务调度,而Spring Batch则专注于批量处理。在这个“quartz_springbatch_dynamic”项目中,我们将看到如何将这两个强大的工具结合起来,以实现动态集群环境中的定时任务执行,并使用MySQL作为数据...

    SpringQuartz的使用文档

    SpringQuartz是一个强大的任务调度框架,它在Java应用程序中用于自动化执行特定任务,如定时生成文件、统计数据或执行后台清理工作。Quartz是开源的,具有高度灵活的调度功能,能够根据预定义的时间表触发任务,同时...

    spring整合quartz定时任务调度

    Spring框架作为Java领域广泛使用的轻量级框架,提供了与第三方库Quartz的整合,使得开发者能够轻松地在Spring应用中实现复杂的定时任务调度。Quartz是一款开源的作业调度框架,支持丰富的调度策略,可以满足各种定时...

    基于Spring Boot的Quartz任务调度系统.zip

    基于Spring Boot的Quartz任务调度系统 项目概述 本项目是一个基于Spring Boot和Quartz的任务调度系统,旨在提供灵活的任务管理和调度功能。系统支持多种任务类型,包括Cron表达式任务和单次时间任务,并提供了丰富...

    Spring中Quartz调度器的使用

    Spring 中 Quartz 调度器的使用 Spring 中 Quartz 调度器的使用是指在 Spring 框架中使用 Quartz 调度器来实现任务调度。Quartz 是一个功能强大且灵活的开源任务调度器,能够满足复杂的任务调度需求。在 Spring 中...

Global site tag (gtag.js) - Google Analytics