1.定义作业调度的底层接口类
package com.job;
/**
* @Package com.buick.activity.job
* @ClassName: JobProcessor
* @Description: 系统作业 接口定义
*/
public interface JobProcessor<T> {
/**
* 作业执行方法
* @return void
* @throws
* @version V1.0
*/
public void execute();
/**
* 作业执行方法
* @return void
* @throws
* @version V1.0
*/
public void executeJob();
/**
* 作业执行状态
* @return String
* @throws
* @version V1.0
*/
public String executeState();
/**
* 作业运行状态
* @return Boolean
* @throws
* @version V1.0
*/
public Boolean runState();
/**
* 启动作业
* @return void
* @throws
* @version V1.0
*/
public void startup();
/**
* 停止作业,不会停止作业中执行的任务
* @return void
* @throws
* @version V1.0
*/
public void shutdown();
}
2.定义作业调度抽象类实现上面接口
package com.job;
import java.sql.Timestamp;
import java.util.Iterator;
import java.util.List;
import java.util.concurrent.atomic.AtomicInteger;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import com.common.util.DateTimeUtils;
/**
* @Package com.buick.activity.job
* @ClassName: BaseJobProcessor
* @Description: 作业逻辑 抽象类
* @author mrajian
* @Version V1.0
* @date 2014-1-26 下午10:29:50
*/
public abstract class BaseJobProcessor<T> implements JobProcessor<T> {
private final static Logger LOGGER = LoggerFactory.getLogger(BaseJobProcessor.class);
// 执行状态
protected Integer executeState = 0;
// 运行状态
protected Boolean runStat = Boolean.TRUE;
//job开始时间
protected Timestamp operationDate;
//job结束时间
protected Timestamp completeDate;
// 真实的任务数量,即查询到的任务数据量
protected AtomicInteger realTaskCount = new AtomicInteger();
protected Integer realTaskSize = 0;
// 执行成功任务数量
protected AtomicInteger successTaskCount = new AtomicInteger();
// 本次作业完成任务数量
protected AtomicInteger completeTaskCount = new AtomicInteger();
/**
* 查询待处理数据的列表
* @return List<T>
* @throws
* @author mrajian
* @date 2014-1-26 下午10:33:02
* @version V1.0
*/
public abstract List<T> queryTaskData();
/**
* 处理待处理数据
* @return void
* @throws
* @author mrajian
* @date 2014-1-26 下午10:33:11
* @version V1.0
*/
public abstract void processTask(T obj);
/**
* JOb数据发生异常时处理
* @return void
* @throws
* @author mrajian
* @date 2014-1-26 下午10:33:23
* @version V1.0
*/
public abstract void queryTaskDataError(Exception e);
/**
* 处理数据时发生异常时的处理
* @param obj
* @return void
* @throws
* @author mrajian
* @date 2014-1-26 下午10:33:36
* @version V1.0
*/
public abstract void processErrorTask(T obj);
/**
* 任务处理完成执行的操作
* @return void
* @throws
* @author mrajian
* @date 2014-1-26 下午10:34:11
* @version V1.0
*/
public abstract void completeTaskProcess();
@Override
public void execute() {
// 用于控制作业的启停
if (!this.runStat) {
return;
}
LOGGER.debug("开始执行作业");
executeJob();
}
@Override
public void executeJob() {
//记录任务开始时间
operationDate = DateTimeUtils.getCurrentTime();
// 批量执行作业
List<T> list = null;
try {
list = queryTaskData();
} catch (Exception e) {
//查询数据出现异常时执行
queryTaskDataError(e);
}
realTaskSize = list == null ? 0 : list.size();
if (list != null && list.size() != 0){
realTaskCount.addAndGet(list.size());
successTaskCount.addAndGet(list.size());
concurrentProcess(list);
}
//记录任务结束时间
completeDate = DateTimeUtils.getCurrentTime();
completeTaskProcess();
}
/**
* 按顺序执行任务
* @param list
* @return void
* @throws
* @author mrajian
* @date 2014-1-26 下午10:40:39
* @version V1.0
*/
private void concurrentProcess(List<T> list) {
Iterator<T> it = list.iterator();
while (it.hasNext()) {
T t = it.next();
try {
processTask(t);
} catch (Exception e) {
LOGGER.error("处理任务出错:",e);
successTaskCount.getAndDecrement();
e.printStackTrace();
try {
processErrorTask(t);
} catch (Exception e1) {
LOGGER.error("处理错误任务出错:",e1);
e1.printStackTrace();
this.completeTaskCount.getAndDecrement();
this.realTaskCount.getAndDecrement();
}
} finally {
this.completeTaskCount.getAndIncrement();
}
}
}
@Override
public String executeState() {
switch(this.executeState){
case 0:
return "未知";
case 1:
return "正常";
default:
return "未知";
}
}
@Override
public Boolean runState() {
return this.runStat;
}
@Override
public void startup() {
this.runStat = Boolean.TRUE;
}
@Override
public void shutdown() {
this.runStat = Boolean.FALSE;
}
}
3.将自己的作业业务逻辑Service继承上面的BaseJobProcessor抽象类4.配置spring-activity-jobs.xml
<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:aop="http://www.springframework.org/schema/aop"
xmlns:tx="http://www.springframework.org/schema/tx" xmlns:context="http://www.springframework.org/schema/context"
xmlns:mvc="http://www.springframework.org/schema/mvc"
xsi:schemaLocation="http://www.springframework.org/schema/beans
http://www.springframework.org/schema/beans/spring-beans-3.1.xsd
http://www.springframework.org/schema/aop
http://www.springframework.org/schema/aop/spring-aop-3.1.xsd
http://www.springframework.org/schema/tx
http://www.springframework.org/schema/tx/spring-tx-3.1.xsd
http://www.springframework.org/schema/jee
http://www.springframework.org/schema/jee/spring-jee-3.1.xsd
http://www.springframework.org/schema/mvc
http://www.springframework.org/schema/mvc/spring-mvc-3.1.xsd
http://www.springframework.org/schema/util
http://www.springframework.org/schema/util/spring-util-3.1.xsd
http://www.springframework.org/schema/context
http://www.springframework.org/schema/context/spring-context-3.1.xsd">
<!-- ======================================================================================= -->
<!-- oss视频转码 任务 测试时间运行 -->
<bean id="transCodeJobSchedulingJobTask" class="org.springframework.scheduling.quartz.MethodInvokingJobDetailFactoryBean">
<property name="targetObject">
<ref bean="transCodeJobService" />
</property>
<property name="targetMethod">
<value>execute</value>
</property>
</bean>
<bean id="transCodeJobSchedulingJobTrigger" class="org.springframework.scheduling.quartz.CronTriggerBean">
<property name="jobDetail">
<ref bean="transCodeJobSchedulingJobTask" />
</property>
<property name="cronExpression">
<value>0 0/1 * ? * *</value>
</property>
</bean>
<!-- ======================================================================================= -->
<!-- 启动调度 -->
<bean autowire="no"
class="org.springframework.scheduling.quartz.SchedulerFactoryBean">
<property name="triggers">
<list>
<ref local="transCodeJobSchedulingJobTrigger" />
</list>
</property>
</bean>
</beans>
<!-- D.CronTrigger配置格式:格式: [秒] [分] [小时] [日] [月] [周] [年]序号 说明 是否必填 允许填写的值
允许的通配符 1 秒 是 0-59 , - * / 2 分 是 0-59 , - * / 3 小时 是 0-23 , - * / 4 日 是 1-31
, - * ? / L W 5 月 是 1-12 or JAN-DEC , - * / 6 周 是 1-7 or SUN-SAT , - * ?
/ L # 7 年 否 empty 或 1970-2099 , - * / 通配符说明: * :表示所有值. 例如:在分的字段上设置 "*",表示每一分钟都会触发。
? :表示不指定值。使用的场景为不需要关心当前设置这个字段的值。例如:要在每月的10号触发一个操作,但不关心是周几,所以需要周位置的那个字段设置为"?"
具体设置为 0 0 0 10 * ?- :表示区间。例如 在小时上设置 "10-12",表示 10,11,12点都会触发。 , :表示指定多个值,例如在周字段上设置
"MON,WED,FRI" 表示周一,周三和周五触发 / :用于递增触发。如在秒上面设置"5/15" 表示从5秒开始,每增15秒触发(5,20,35,50)。
在月字段上设置'1/3'所示每月1号开始,每隔三天触发一次。 L :表示最后的意思。在日字段设置上,表示当月的最后一天(依据当前月份,如果是二月还会依据是否是润年[leap]),
在周字段上表示星期六,相当于"7"或"SAT"。如果在"L"前加上数字,则表示该数据的最后一个。例如在周字段上设置"6L"这样的格式,则表示“本月最后一个星期五"
W :表示离指定日期的最近那个工作日(周一至周五). 例如在日字段上设置"15W",表示离每月15号最近的那个工作日触发。如果15号正好是周六,则找最近的周五(14号)触发,
如果15号是周未,则找最近的下周一(16号)触发.如果15号正好在工作日(周一至周五),则就在该天触发。如果指定格式为 "1W",它则表示每月1号往后最近的工作日触发。如果1号正是周六,则将在3号下周一触发。(注,"W"前只能设置具体的数字,不允许区间"-").'L'和
'W'可以一组合使用。如果在日字段上设置"LW",则表示在本月的最后一个工作日触发 # :序号(表示每月的第几周星期几),例如在周字段上设置"6#3"表示在每月的第三个周星期六.注意如果指定"6#5",正好第五周没有星期六,则不会触发该配置(用在母亲节和父亲节再合适不过了)
周字段的设置,若使用英文字母是不区分大小写的 MON 与mon相同.常用示例: 格式: [秒] [分] [小时] [日] [月] [周] [年]
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分每分触发 0 0/5 14 * * ? 每天下午的 2点到2点59分(整点开始,每隔5分触发) 0 0/5 14,18
* * ? 每天下午的 18点到18点59分(整点开始,每隔5分触发) 0 0-5 14 * * ? 每天下午的 2点到2点05分每分触发 0 10,44
14 ? 3 WED 3月分每周三下午的 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 每月的第三周的星期五开始触发 0 0 12 1/5 * ? 每月的第一个中午开始每隔5天触发一次 0 11 11
11 11 ? 每年的11月11号 11点11分触发(光棍节)注意:使用SimpleTriggerBean类只能做简单Job与Job之间的执行周期指定,如果需要在指定时间执行,可以使用CronTriggerBean类。
-->