【spring配置】——spring整合Quartz定时器
第一种:为普通java类中的某个方法配置跑批任务
- MethodInvokingJobDetailFactoryBean
- CronTriggerBean
- SchedulerFactoryBean
1.定义要跑批的类和方法:
- package com.xy.utils.quartz;
- import org.joda.time.DateTime;
- import org.slf4j.Logger;
- import org.slf4j.LoggerFactory;
- /**
- * 测试定时器类
- * @author javaw
- *
- */
- public class TestQuartz {
- public static Logger logger = LoggerFactory.getLogger(TestQuartz.class);
- public void TestMethod(){
- logger.info("Auto Execute TestMethod start! Date={}" ,new DateTime().toString("YYYY-MM-DD HH:mm:ss" ));
- logger.info("**********测试跑批类************");
- logger.info("Auto Execute TestMethod end! Date={}" ,new DateTime().toString("YYYY-MM-DD HH:mm:ss" ));
- }
- }
2.配置Spring定时器让quartz自动执行testMethod方法:
- <!-- 实例化bean -->
- <bean id= "testMethodQuartz" class ="com.xy.utils.quartz.TestQuartz"/>
- <!-- 配置MethodInvokingJobDetailFactoryBean -->
- <bean id= "testTaskMethod"
- class="org.springframework.scheduling.quartz.MethodInvokingJobDetailFactoryBean">
- <property name="targetObject" ref="testMethodQuartz"/>
- <property name="targetMethod" value="TestMethod"/>
- <property name="concurrent" value="false"/>
- </bean>
- <!-- 配置定时表达式 -->
- <bean id= "testTaskTrigger" class="org.springframework.scheduling.quartz.CronTriggerBean" >
- <property name="jobDetail" ref="testTaskMethod" />
- <!-- 每一分钟执行一次 -->
- <property name="cronExpression" value="0 0/1 * * * ?" />
- </bean>
- <!-- 配置调度工厂 -->
- <bean id= "testSchedulerFactoryBean"
- class="org.springframework.scheduling.quartz.SchedulerFactoryBean">
- <property name="triggers" >
- <list>
- <ref bean="testTaskTrigger" />
- </list>
- </property>
- </bean>
3.写测试方法进行调用(不用启动tomcat):
- public static void main(String[] args) {
- SchedulerFactoryBean schedulerFactoryBean = (SchedulerFactoryBean) SpringUtilsFromClassPathXml.getBean( "testSchedulerFactoryBean");
- //启动调度器
- schedulerFactoryBean.start();
- }
第二种:为继承QuartzJobBean的java类配置跑批任务
1.写java类继承QuartzJobBean,重写executeInternal 方法:
- package com.xy.utils.quartz;
- import org.joda.time.DateTime;
- import org.quartz.JobExecutionContext;
- import org.quartz.JobExecutionException;
- import org.slf4j.Logger;
- import org.slf4j.LoggerFactory;
- import org.springframework.scheduling.quartz.QuartzJobBean;
- import org.springframework.scheduling.quartz.SchedulerFactoryBean;
- /**
- * 测试继承QuartzJobBean的java类配置定时器
- * @author javaw
- *
- */
- public class TestQuartzExtendsJobBean extends QuartzJobBean{
- private static Logger logger = LoggerFactory.getLogger(TestQuartzExtendsJobBean. class);
- @Override
- protected void executeInternal(JobExecutionContext context)
- throws JobExecutionException {
- logger.info("Auto Execute TestQuartzExtendsJobBean start! Date={}",new DateTime().toString( "YYYY-MM-DD HH:mm:ss"));
- logger.info("**********"+content.getMergedJobDataMap().get("descString")+"************");
- logger.info("Auto Execute TestQuartzExtendsJobBean end! Date={}",new DateTime().toString( "YYYY-MM-DD HH:mm:ss"));
- }
- }
2.配置文件:
- <!--
- 第二种:为继承Quartz的java类实现跑批
- **********1.JobDetailBean
- **********2.CronTriggerBean
- **********3.SchedulerFactoryBean
- -->
- <bean id= "testJobDetailBean"
- class="org.springframework.scheduling.quartz.JobDetailBean">
- <property name="name" value="exampleJob" />
- <property name="jobClass"
- value="com.xy.utils.quartz.TestQuartzExtendsJobBean"/>
- <!-- 可以封装各种数据到JobExecutionContext里 -->
- <property name="jobDataAsMap">
- <map>
- <entry key="descString" value="测试跑批"/>
- </map>
- </property>
- </bean >
- <bean id= "testTaskTrigger"
- class="org.springframework.scheduling.quartz.CronTriggerBean">
- <property name="jobDetail" ref="testJobDetailBean" />
- <property name="cronExpression" value="0 0/1 * * * ?" />
- </bean>
- <bean id= "testSchedulerFactoryBean"
- class="org.springframework.scheduling.quartz.SchedulerFactoryBean">
- <property name="triggers" >
- <list>
- <ref bean="testTaskTrigger" />
- </list>
- </property>
- </bean>
第三种:定时任务持久化(JobDetailTx)
优势:
集群部署定时器通过故障切换和负载均衡的功能,提高调度器的可用性和扩展性。
本质:
集群上的所有节点通过共享一个数据库来来工作的。所有节点quartz通过启动两个维护线程来维护数据库状态实现集群管理,一个是检测节点状态线程,一个是恢复任务线程。一个 Quartz 集群中的每个节点是一个独立的 Quartz 应用,它又管理着其他的节点。
模式:
1.获取任务
负载均衡是自动完成的,集群的每个节点会尽快触发任务。当第一个节点获取到任务,会通过锁定,阻止其他线程获取到该任务。
2.故障切换
当一个节点执行任务失败(服务down掉或其他原因),其他节点会检测到并标识在失败节点上正在执行的数据库中的任务。任何标记为可恢复(任务详细里的requests recovery属性)的任务都会被其他节点继续执行。没有标记可恢复的任务会被释放掉。
- CREATE TABLE qrtz_job_details
- (
- ....
- JOB_NAME VARCHAR2(80) NOT NULL,
- JOB_GROUP VARCHAR2(80) NOT NULL,
- REQUESTS_RECOVERY VARCHAR2(1) NOT NULL, --可恢复标记
- );
- SchedulerFactoryBean
- JobDetail
- Trigger
- Job
将定时任务持久化可以解决分布式跑批的问题,避免服务器重启信息丢失。
定时任务定义需要实例化JobDetail和Trigger,然后使用Scheduler进行调度。这个需要使用到Quartz的jar包。
每个jar包创建任务的方式都不相同。具体需要查看官方api。
quartz2.2.1中给出的示例:
- JobDetail job = newJob(MyJob.class)
- .withIdentity("myJob")
- .build();
- Trigger trigger = newTrigger()
- .withIdentity(triggerKey("myTrigger", "myTriggerGroup"))
- .withSchedule(simpleSchedule()
- .withIntervalInHours(1)
- .repeatForever())
- .startAt(futureDate(10, MINUTES))
- .build();
- scheduler.scheduleJob(job, trigger);
导入jar包后,写如下测试类:
- package com.xy.utils.scheduler;
- import org.quartz.JobBuilder;
- import org.quartz.JobDetail;
- import org.quartz.Scheduler;
- import org.quartz.SchedulerFactory;
- import org.quartz.SimpleScheduleBuilder;
- import org.quartz.Trigger;
- import org.quartz.TriggerBuilder;
- import org.quartz.impl.StdSchedulerFactory;
- public class quartzTest {
- public static void main(String args[]) throws Exception {
- JobDetail jobDetail= JobBuilder.newJob(TestJob.class)
- .withIdentity("job","group")
- .build();
- Trigger trigger= TriggerBuilder.newTrigger().withIdentity("job","group").startNow().withSchedule(
- SimpleScheduleBuilder.simpleSchedule()
- .withIntervalInSeconds(10) //时间间隔
- .withRepeatCount(10) //重复次数n+1
- )
- .build();
- SchedulerFactory sf = new StdSchedulerFactory();
- Scheduler sched = sf.getScheduler();
- sched.scheduleJob(jobDetail, trigger);
- sched.start();
- }
- }
定时执行的任务类:
- package com.xy.utils.scheduler;
- import org.quartz.Job;
- import org.quartz.JobExecutionContext;
- import org.quartz.JobExecutionException;
- public class TestJob implements Job {
- public void execute(JobExecutionContext arg0) throws JobExecutionException {
- System.out.println("*****执行批处理任务******");
- }
- }
这种任务并没有持久化到数据库中。
在Spring中配置SchedulerFactoryBean:
- <!--
- 第三种:从数据库中读取跑批任务(适合集群部署跑批)
- **********1.SchedulerFactoryBean
- -->
- <bean id="schedulerBeanFactory" class="org.springframework.scheduling.quartz.SchedulerFactoryBean">
- <!-- 注入数据源,包含任务执行表 -->
- <property name="dataSource" ref="dataSource"/>
- <!--applicationContextSchedulerContextKey:
- 把spring上下文以key/value的方式存放在了quartz的上下文中了,
- 可以用applicationContextSchedulerContextKey所定义的key得到对应的spring上下文-->
- <property name="applicationContextSchedulerContextKey" value="applicationContext"/>
- <!-- 读取配置文件 -->
- <property name="configLocation" value="classpath:/properties/quartz.properties"/>
- <property name="autoStartup" value="true"/>
- </bean>
配置quartz.properties:
下载了quartz.jar后,在\quartz-2.2.1\src\org\quartz\目录下有个quartz.properties的文件,可以根据自己的需要更改其中的配置内容。
- #批处理常量表 QUARTZ2.2.1
- #============================================================================
- # Configure Main Scheduler Properties
- #============================================================================
- org.quartz.scheduler.instanceName = MyQuartzScheduler
- org.quartz.scheduler.instanceId = AUTO
- #============================================================================
- # Configure ThreadPool
- #============================================================================
- org.quartz.threadPool.class = org.quartz.simpl.SimpleThreadPool
- org.quartz.threadPool.threadCount = 10
- org.quartz.threadPool.threadPriority = 5
- org.quartz.threadPool.threadsInheritContextClassLoaderOfInitializingThread:true
- #============================================================================
- # Configure JobStore
- #============================================================================
- org.quartz.jobStore.misfireThreshold = 10000
- org.quartz.jobStore.class = org.springframework.scheduling.quartz.LocalDataSourceJobStore
- org.quartz.jobStore.driverDelegateClass = org.quartz.impl.jdbcjobstore.StdJDBCDelegate
- org.quartz.jobStore.isClustered = true --是否集群部署
- #任务表前缀
- org.quartz.jobStore.tablePrefix = test_qrtz_
创建任务表:
jar包文件中,给出了所有数据库的建表语句。可以根据给出的sql语句创建表。
调度类:
- package com.xy.utils.scheduler;
- import org.quartz.JobBuilder;
- import org.quartz.JobDetail;
- import org.quartz.Scheduler;
- import org.quartz.SimpleScheduleBuilder;
- import org.quartz.Trigger;
- import org.quartz.TriggerBuilder;
- import com.xy.utils.SpringUtils.SpringUtilsFromClassPathXml;
- public class quartzTest {
- //从Spring中获取调度bean工厂
- public static Scheduler scheduler = (Scheduler)SpringUtilsFromClassPathXml.getBean("schedulerBeanFactory");
- public static void main(String args[]) throws Exception {
- JobDetail jobDetail= JobBuilder.newJob(TestJob.class)
- .withIdentity("job","group")
- .build();
- Trigger trigger= TriggerBuilder.newTrigger().withIdentity("job","group").startNow().withSchedule(
- SimpleScheduleBuilder.simpleSchedule()
- .withIntervalInSeconds(10) //时间间隔
- .repeatForever()
- )
- .build();
- scheduler.scheduleJob(jobDetail, trigger);
- scheduler.start();
- }
- }
任务类(需要实现job接口):
- package com.xy.utils.scheduler;
- import org.quartz.Job;
- import org.quartz.JobExecutionContext;
- import org.quartz.JobExecutionException;
- public class TestJob implements Job {
- public void execute(JobExecutionContext arg0) throws JobExecutionException {
- System.out.println("*****执行批处理任务******");
- }
- }
此时执行该类就会将任务数据写入数据库中。
调度任务工具类:
自定义任务调度工具类:
- package com.xy.utils.scheduler;
- import java.text.ParseException;
- import java.text.SimpleDateFormat;
- import java.util.Date;
- import org.quartz.CronScheduleBuilder;
- import org.quartz.CronTrigger;
- import org.quartz.DateBuilder;
- import org.quartz.Job;
- import org.quartz.JobBuilder;
- import org.quartz.JobDetail;
- import org.quartz.Scheduler;
- import org.quartz.SchedulerException;
- import org.quartz.SimpleScheduleBuilder;
- import org.quartz.Trigger;
- import org.quartz.TriggerBuilder;
- import org.quartz.TriggerKey;
- import org.quartz.DateBuilder.IntervalUnit;
- import org.slf4j.Logger;
- import org.slf4j.LoggerFactory;
- import com.xy.common.SysContent;
- public class SchedulerJobUtils {
- private static Logger logger = LoggerFactory.getLogger(SchedulerJobUtils.class);
- public static Scheduler scheduler = (Scheduler)SysContent.applicationContext.getBean("schedulerBeanFactory");
- //public static Scheduler scheduler = (Scheduler)SpringUtilsFromClassPathXml.getBean("schedulerBeanFactory");
- public static String dateFormate = "ss mm HH dd MM ? yyyy";
- public static SimpleDateFormat cronSdf = new SimpleDateFormat(dateFormate);
- public static SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");
- /**
- * 数据库放入一个指定时间的任务,任务只执行一次
- * @see 适合对数据进行定时删除,修改等等
- * @param <T>
- * @param clazz 要执行任务类class,任务类必须继承Job类
- * @param jobName 任务名称
- * @param groupName 分组名称
- * @param executeTime 执行时间,格式:yyyy-MM-dd HH:mm:ss
- * @return
- */
- public static<T> boolean putSpecifyTimeSchedulerJobToDB(Class<? extends Job> clazz,String jobName,String groupName,String executeTime){
- //定义一个job
- JobDetail job = JobBuilder.newJob(clazz).withIdentity(jobName, groupName).build();
- //定义一个TriggerKey
- TriggerKey triggerKey = TriggerKey.triggerKey(jobName, groupName);
- SchedulerJob schedulerJob = new SchedulerJob();
- schedulerJob.setJobName(jobName);
- schedulerJob.setGroupName(groupName);
- job.getJobDataMap().put("schedulerJob", schedulerJob);
- Date executeDate = null;
- try {
- executeDate = sdf.parse(executeTime);
- } catch (ParseException e) {
- logger.info("method putSchedulerJobToDB execute error!exception={}",e);
- }
- String dbExecuteTime = cronSdf.format(executeDate);
- CronTrigger cronTrigger = TriggerBuilder.newTrigger().withIdentity(triggerKey).withSchedule(CronScheduleBuilder.cronSchedule(dbExecuteTime)).build();
- try {
- scheduler.scheduleJob(job, cronTrigger);
- } catch (SchedulerException e) {
- logger.info("method putSchedulerJobToDB execute error!exception={}",e);
- }
- return true;
- }
- /**
- * 数据库放入一条指定开始时间,指定间隔时间,指定次数的任务
- * @param <T>
- * @param clazz 任务类
- * @param jobName 任务名
- * @param groupName 分组名
- * @param seconds 间隔时间,以秒为单位
- * @param count 执行次数,0为一直执行
- * @param startSecondsToNow 开始时间(距离现在?秒)
- * @return
- * @throws SchedulerException
- */
- @SuppressWarnings("unchecked")
- public static<T> boolean putPeriodSchedulerJobToDB(Class<? extends Job> clazz,String jobName,String groupName,
- int seconds,int count,int startSecondsToNow){
- JobDetail jobDetail= JobBuilder.newJob(clazz).withIdentity(jobName,groupName).build();
- TriggerBuilder triggerBuilder = TriggerBuilder.newTrigger().withIdentity(jobName,groupName);
- if(startSecondsToNow==0){
- triggerBuilder.startNow();
- }else{
- triggerBuilder.startAt(DateBuilder.futureDate(startSecondsToNow,IntervalUnit.SECOND));
- }
- SimpleScheduleBuilder scheduleBuilder = SimpleScheduleBuilder.simpleSchedule().withIntervalInSeconds(seconds);
- if(count==0){
- scheduleBuilder.repeatForever();
- }else{
- scheduleBuilder.withRepeatCount(count);
- }
- Trigger trigger = triggerBuilder.withSchedule(scheduleBuilder).build();
- try {
- scheduler.scheduleJob(jobDetail,trigger);
- } catch (SchedulerException e) {
- logger.info("put putPeriodSchedulerJobToDB error,error={}",e);
- }
- return true;
- }
- }
任务类:
- package com.xy.utils.scheduler;
- import org.joda.time.DateTime;
- import org.quartz.Job;
- import org.quartz.JobExecutionContext;
- import org.quartz.JobExecutionException;
- import org.slf4j.Logger;
- import org.slf4j.LoggerFactory;
- public class TestSchedulerTask implements Job{
- private static Logger logger = LoggerFactory.getLogger(TestSchedulerTask.class);
- public void execute(JobExecutionContext context) throws JobExecutionException {
- logger.info("**********TestSchedulerTask1 start at={}",new DateTime().toString("yyyy-MM-dd HH:mm:ss"));
- logger.info("**********I'm Ok!************");
- logger.info("**********TestSchedulerTask1 end at={}",new DateTime().toString("yyyy-MM-dd HH:mm:ss"));
- }
- /**
- * 调度任务工具类测试
- * 10秒执行一次,无限次数,30秒后开始
- * @param args
- */
- public static void main(String[] args) {
- System.out.println(SchedulerJobUtils.putPeriodSchedulerJobToDB(TestSchedulerTask.class, "myJob", "myGroup", 10, 0, 30));
- }
- }
数据库里的任务表:
打印出的日志:
2016-01-10 00:37:34,810 INFO TestSchedulerTask1:14 - **********TestSchedulerTask1 start at=2016-01-10 00:37:34
2016-01-10 00:37:34,810 INFO TestSchedulerTask1:15 - **********I'm Ok!************
2016-01-10 00:37:34,811 INFO TestSchedulerTask1:16 - **********TestSchedulerTask1 end at=2016-01-10 00:37:34
2016-01-10 00:37:44,690 INFO TestSchedulerTask1:14 - **********TestSchedulerTask1 start at=2016-01-10 00:37:44
2016-01-10 00:37:44,691 INFO TestSchedulerTask1:15 - **********I'm Ok!************
2016-01-10 00:37:44,691 INFO TestSchedulerTask1:16 - **********TestSchedulerTask1 end at=2016-01-10 00:37:44
2016-01-10 00:37:54,697 INFO TestSchedulerTask1:14 - **********TestSchedulerTask1 start at=2016-01-10 00:37:54
2016-01-10 00:37:54,697 INFO TestSchedulerTask1:15 - **********I'm Ok!************
2016-01-10 00:37:54,697 INFO TestSchedulerTask1:16 - **********TestSchedulerTask1 end at=2016-01-10 00:37:54
【spring配置】——spring整合Quartz定时器的更多相关文章
- Spring Boot 整合Quartz定时器
概述 项目需要定时器的调度管理,原来使用Spring Boot自带的定时器,但是不能后台动态的操作暂停.启动以及新增任务等操作,维护起来相对麻烦:最近研究了Quartz的框架,觉得还算不错,整理了一下 ...
- spring通过配置xml文件集成quartz定时器
概述 Spring为创建Quartzde Scheduler.Trigger和JobDetail提供了方便的FactoryBean类,以便能够在Spring容器中享受注入的好处. 此外,Spring还 ...
- Spring整合Quartz定时器
1.添加jar #此处省略spring核心jar包 <dependency> <groupId>org.quartz-scheduler</groupId> < ...
- Spring Boot 应用系列 6 -- Spring Boot 2 整合Quartz
Quartz是实现定时任务的利器,Quartz主要有四个组成部分,分别是: 1. Job(任务):包含具体的任务逻辑: 2. JobDetail(任务详情):是对Job的一种详情描述: 3. Trig ...
- spring data JPA使用quartz定时器的具体实现
第一步.在pom.xml中的配置 <!--quartz--> <dependency> <groupId>org.quartz-scheduler</grou ...
- Spring 配置 Spring JPA 发生错误的解决方法
今天在项目的applicationContext.xml中配JPA时 <?xml version="1.0" encoding="UTF-8"?> ...
- Spring整合Quartz实现动态定时器
一.版本说明 spring3.1以下的版本必须使用quartz1.x系列,3.1以上的版本才支持quartz 2.x,不然会出错. 原因:spring对于quartz的支持实现,org.springf ...
- Spring整合Quartz实现动态定时器,相关api,定时器添加,删除,修改
一.版本说明 spring3.1以下的版本必须使用quartz1.x系列,3.1以上的版本才支持quartz 2.x,不然会出错. 原因:spring对于quartz的支持实现,org.springf ...
- Spring 整合 Quartz框架(定时任务)
Maven 无法下载 Quartz 依赖,去官网下载 http://www.quartz-scheduler.org/downloads/ Quartz 官方手册:https://www.w3csch ...
随机推荐
- Codeforces 785D Anton and School - 2(推公式+乘法原理+组合数学)
题目链接 Anton and School - 2 对于序列中的任意一个单括号对(), 左括号左边(不含本身)有a个左括号,右括号右边(不含本身有)b个右括号. 那么答案就为 但是这样枚举左右的()的 ...
- [Python Cookbook] Numpy: Iterating Over Arrays
1. Using for-loop Iterate along row axis: import numpy as np x=np.array([[1,2,3],[4,5,6]]) for i in ...
- POJ 1704 Georgia and Bob [博弈]
题意:一个仅有一行的棋盘上,初始时有n个棋子,每人轮流移动棋子,每次只能移动一枚棋子,棋子在移动时只能向左移动,不能跨过别的棋子或跳出棋盘. 思路:这道题是一道nim游戏的巧妙变形,太棒了. 解决的思 ...
- group by与distinct效率分析及优化措施
如何使用group by进行去重 因为mysql的distinct在结果集中,全部不同,才可以去重.所以,当我们进行去重处理的时候,需要单独对某列进行去重,可以使用group by子句进行分组去重se ...
- 黑苹果+win10双系统折腾笔记
寒假趁机在家折腾一下黑苹果 笔记本配置:神船K610D I7 4600 ,其他配置思路一样,驱动要自己找 镜像和工具:OS X Yosemite 10.10.3 镜像 WIN10 TLSB 2016 ...
- 基于Wiremock创建Mock Service平台
转载:http://blog.csdn.net/liuchunming033/article/details/52399397 1.Wiremock工具介绍 一般开发项目都会分模块进行,比如都会把前端 ...
- mongodb管理副本集(持续更新中)
许多维护工作不能在备份节点上完成 因为要写操作,也不能在主节点上进行,这就需要单机模式启动服务器, 是指重启成员服务器,让他成为一个单机运行的服务器,而不再是副本集中的一员(临时的) 在单机 ...
- element的el-tabs控制,以及el-select 多选默认值
一.el-tabs 1.element自己已经封装好了,当切换时v-model的值自动切换为el-tabs-pane的name对应的值. 如下: <el-tabs v-model='active ...
- 关于郭天祥51开发板无法烧敲代码问题的解决(Prolific USB-to-Serial Comm Port)
1. 事件背景: 因为使用了win8系统,之前购买的郭天祥C51开发板在通过一个两头都是usb口的下载线下载程序时出现了问题:下载工具stc isp无法连接到开发板上的串口,所以无法下载程序到c51开 ...
- python:如何判断字符串中的内容是否都为数字并且把字符串转换为数字
使用str.isdigit();有两种使用方法 str.isdigit('12345') =====>True str.isdigit('aaaaa')======>False 或者 '1 ...