<?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:context="http://www.springframework.org/schema/context"
xsi:schemaLocation="http://www.springframework.org/schema/beans
http://www.springframework.org/schema/beans/spring-beans-4.3.xsd
http://www.springframework.org/schema/context
http://www.springframework.org/schema/context/spring-context-4.3.xsd">
<context:component-scan base-package="com.bao"/>
<!-- 引入外部数据源配置信息 -->
<bean class="org.springframework.beans.factory.config.PropertyPlaceholderConfigurer">
<property name="locations">
<list>
<value>classpath:${env.active}/quartz.properties</value>
<value>classpath:${env.active}/jdbc.properties</value>
</list>
</property>
<property name="ignoreUnresolvablePlaceholders" value="true"/>
</bean>
<bean id="dataSource" class="com.alibaba.druid.pool.DruidDataSource" destroy-method="close">
<!-- 数据库基本信息配置 -->
<property name="url" value="${jdbc.url}"/>
<property name="username" value="${jdbc.username}"/>
<property name="password" value="${jdbc.password}"/>
<property name="driverClassName" value="${jdbc.driver}"/>
<property name="filters" value="${filters}"/>
<!-- 最大并发连接数 -->
<property name="maxActive" value="${maxActive}"/>
<!-- 初始化连接数量 -->
<property name="initialSize" value="${initialSize}"/>
<!-- 配置获取连接等待超时的时间 -->
<property name="maxWait" value="${maxWait}"/>
<!-- 最小空闲连接数 -->
<property name="minIdle" value="${minIdle}"/>
<!-- 配置间隔多久才进行一次检测,检测需要关闭的空闲连接,单位是毫秒 -->
<property name="timeBetweenEvictionRunsMillis" value="${timeBetweenEvictionRunsMillis}"/>
<!-- 配置一个连接在池中最小生存的时间,单位是毫秒 -->
<property name="minEvictableIdleTimeMillis" value="${minEvictableIdleTimeMillis}"/>
<property name="validationQuery" value="${validationQuery}"/>
<property name="testWhileIdle" value="${testWhileIdle}"/>
<property name="testOnBorrow" value="${testOnBorrow}"/>
<property name="testOnReturn" value="${testOnReturn}"/>
<property name="maxOpenPreparedStatements" value="${maxOpenPreparedStatements}"/>
<!-- 打开 removeAbandoned 功能 -->
<property name="removeAbandoned" value="${removeAbandoned}"/>
<!-- 1800 秒 -->
<property name="removeAbandonedTimeout" value="${removeAbandonedTimeout}"/>
<!-- 关闭 abanded 连接时输出错误日志 -->
<property name="logAbandoned" value="${logAbandoned}"/>
</bean>
<bean id="getSchedulerFactoryBean" class="org.springframework.scheduling.quartz.SchedulerFactoryBean"
lazy-init="false">
<property name="autoStartup" value="true"/>
<property name="dataSource" ref="dataSource"/>
<property name="triggers">
<list>
<!--list调度器-->
<!--<ref bean="helloJobCronTriggerBean" />-->
</list>
</property>
</bean>
<bean id="dynamicSchedulerFactory" class="com.*.core.cfg.quartz.DynamicSchedulerFactory">
<property name="scheduler" ref="getSchedulerFactoryBean"/>
</bean>
<!--示例-->
<!--<bean id="helloJobDetail"-->
<!--class="org.springframework.scheduling.quartz.JobDetailFactoryBean">-->
<!--<property name="jobClass" value="com.swxc.app.job.HelloJob" />-->
<!--<property name="durability" value="true" />-->
<!--</bean>-->
<!--<bean id="helloJobCronTriggerBean"-->
<!--class="org.springframework.scheduling.quartz.CronTriggerFactoryBean">-->
<!--<property name="jobDetail" ref="helloJobDetail" />-->
<!--<property name="cronExpression" value="0 * * * * ?" />-->
<!--<property name="misfireInstruction" value="2" />-->
<!--</bean>-->
</beans>
filters=stat
maxActive=20
initialSize=5
maxWait=60000
minIdle=10
maxIdle=15
timeBetweenEvictionRunsMillis=60000
minEvictableIdleTimeMillis=300000
validationQuery=SELECT 'x'
testWhileIdle=true
testOnBorrow=false
testOnReturn=false
maxOpenPreparedStatements=20
removeAbandoned=true
removeAbandonedTimeout=1800
logAbandoned=true
import org.quartz.*;
import java.util.Map;
public class DynamicJob {
//要执行类, 实现Job接口
private Class<? extends Job> target;
//cron 表达式
private String cronExpression;
private String jobGroup = Scheduler.DEFAULT_GROUP;
//必须唯一
private String jobName;
private transient TriggerKey triggerKey;
private transient JobDetail jobDetail;
//default
public DynamicJob() {
}
public DynamicJob(String jobName) {
this.jobName = jobName;
}
public Class<? extends Job> target() {
return target;
}
public DynamicJob target(Class<? extends Job> target) {
this.target = target;
return this;
}
public DynamicJob cronExpression(String cronExpression) {
this.cronExpression = cronExpression;
return this;
}
public String jobGroup() {
return jobGroup;
}
public DynamicJob jobGroup(String jobGroup) {
this.jobGroup = jobGroup;
return this;
}
public String jobName() {
return jobName;
}
public DynamicJob jobName(String jobName) {
this.jobName = jobName;
return this;
}
public TriggerKey triggerKey() {
if (triggerKey == null) {
triggerKey = TriggerKey.triggerKey(this.jobName, this.jobGroup);
}
return triggerKey;
}
public JobDetail jobDetail() {
if (jobDetail == null) {
jobDetail = JobBuilder.newJob(target)
.withIdentity(this.jobName, this.jobGroup)
.build();
}
return jobDetail;
}
/*
* 传参数给 执行的 job
* 在job中 通过
* context.getMergedJobDataMap().get(key) 获取值
* */
public DynamicJob addJobData(String key, Object value) {
final JobDetail detail = jobDetail();
final JobDataMap jobDataMap = detail.getJobDataMap();
jobDataMap.put(key, value);
return this;
}
/*
* 传参数给 执行的 job
* 在job中 通过
* context.getMergedJobDataMap().get(key) 获取值
* */
public DynamicJob addJobDataMap(Map<String, Object> map) {
final JobDetail detail = jobDetail();
final JobDataMap jobDataMap = detail.getJobDataMap();
jobDataMap.putAll(map);
return this;
}
public String cronExpression() {
return this.cronExpression;
}
@Override
public String toString() {
final StringBuilder sb = new StringBuilder();
sb.append("{target=").append(target);
sb.append(", cronExpression='").append(cronExpression).append('\'');
sb.append(", jobGroup='").append(jobGroup).append('\'');
sb.append(", jobName='").append(jobName).append('\'');
sb.append('}');
return sb.toString();
}
}
import org.quartz.*;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.beans.factory.InitializingBean;
import org.springframework.util.Assert;
import java.util.Date;
public final class DynamicSchedulerFactory implements InitializingBean {
private static final Logger LOG = LoggerFactory.getLogger(DynamicSchedulerFactory.class);
private static Scheduler scheduler;
// private static SchedulerFactory scheduler = new StdSchedulerFactory();
public DynamicSchedulerFactory() {
}
/**
* 注册一个job
* 1.检查是否有相同的 scheduler(根据 trigger key) ,若有则抛出异常
* 2.调用 {@link #scheduler} 的 scheduleJob 加入
*
* @param job DynamicJob
* @return True is register successful
* @throws org.quartz.SchedulerException
*/
public static boolean registerJob(DynamicJob job) throws SchedulerException {
final TriggerKey triggerKey = job.triggerKey();
if (scheduler.checkExists(triggerKey)) {
final Trigger trigger = scheduler.getTrigger(triggerKey);
throw new SchedulerException("Already exist trigger [" + trigger + "] by key [" + triggerKey + "] in Scheduler");
}
final CronScheduleBuilder cronScheduleBuilder = CronScheduleBuilder.cronSchedule(job.cronExpression());
final CronTrigger cronTrigger = TriggerBuilder.newTrigger().withIdentity(triggerKey)
.withSchedule(cronScheduleBuilder)
.build();
final JobDetail jobDetail = job.jobDetail();
final Date date = scheduler.scheduleJob(jobDetail, cronTrigger);
LOG.debug("Register DynamicJob {} on [{}]", job, date);
return true;
}
/**
* 删除掉已经存在的 job
*
* @param existJob A DynamicJob which exists in Scheduler
* @return True is remove successful
* @throws org.quartz.SchedulerException
*/
public static boolean removeJob(DynamicJob existJob) throws SchedulerException {
final TriggerKey triggerKey = existJob.triggerKey();
boolean result = false;
if (scheduler.checkExists(triggerKey)) {
result = scheduler.unscheduleJob(triggerKey);
}
LOG.debug("Remove DynamicJob {} result [{}]", existJob, result);
return result;
}
public void setScheduler(Scheduler scheduler) {
DynamicSchedulerFactory.scheduler = scheduler;
}
@Override
public void afterPropertiesSet() throws Exception {
Assert.notNull(scheduler, "scheduler is null");
LOG.info("Initial DynamicSchedulerFactory successful, scheduler instance: {}", scheduler);
}
}
import org.quartz.*;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import java.util.Date;
@PersistJobDataAfterExecution
@DisallowConcurrentExecution
public class HelloJob implements Job {
private static Logger _log = LoggerFactory.getLogger(HelloJob.class);
public HelloJob() {
}
@Override
public void execute(JobExecutionContext context) throws JobExecutionException {
_log.error("Hello Job执行时间: " + new Date());
System.out.println("Hello Job执行时间: " + new Date());
}
}
@org.springframework.stereotype.Controller
@RequestMapping(value = "/test")
public class TestController extends Controller {
@RequestMapping("/add/job")
public Json addJob() {
try {
DynamicJob job = new DynamicJob(BaseUtil.uuid())
.cronExpression("0 */1 * * * ?")
.target(HelloJob.class)
.addJobData("_ID", 1);
DynamicSchedulerFactory.registerJob(job);
} catch (SchedulerException e) {
e.printStackTrace();
throw new BizException("添加失败");
}
return json(BizUtils.SUCCESS_CODE, BizUtils.SUCCESS_MESSAGE);
}
}
<dependency>
<groupId>org.quartz-scheduler</groupId>
<artifactId>quartz</artifactId>
<version>2.2.1</version>
<scope>compile</scope>
<exclusions>
<exclusion>
<artifactId>slf4j-api</artifactId>
<groupId>org.slf4j</groupId>
</exclusion>
<exclusion>
<artifactId>c3p0</artifactId>
<groupId>c3p0</groupId>
</exclusion>
</exclusions>
</dependency>
<dependency>
<groupId>com.alibaba</groupId>
<artifactId>druid</artifactId>
<version>1.1.10</version>
</dependency>