Spring_任务调度的简单笔记

本文介绍了一个自定义的作业调度框架的设计与实现过程,包括定义作业调度接口、抽象类及具体的业务逻辑实现,并通过Spring框架进行配置,实现了定时作业的自动化执行。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

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类。
 -->


评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值