由于项目中定时任务逐渐增多,对系统的压力也慢慢增加。故打算将系统中的定时任务抽离出来。初步决定使用 SpringBoot+mybatis+quartz 的整合方式进行快速开发。
整个整合会包含如下任务:
<dependency>
<groupId>org.mybatis.spring.bootgroupId>
<artifactId>mybatis-spring-boot-starterartifactId>
<version>${mybatis-spring-boot-starter.version}version>
dependency>
<dependency>
<groupId>mysqlgroupId>
<artifactId>mysql-connector-javaartifactId>
<scope>runtimescope>
dependency>
主要添加数据库连接配置和mybatis的mapper的xml配置文件路径以及实体类的包。还有一些mybatis的相关配置:mybatis相关配置参数参考
# 配置数据库连接
spring.datasource.driver-class-name = com.mysql.jdbc.Driver
spring.datasource.url = jdbc:mysql://localhost:3306/taskmgr?useUnicode=true&characterEncoding=utf-8&useSSL=false
spring.datasource.username = root
spring.datasource.password = 123456
#mybatis
mybatis.type-aliases-package=com.pingan.wechat.app.entity
mybatis.mapper-locations=classpath:mapper/*.xml
#more configuration about mybatis : http://www.mybatis.org/mybatis-3/zh/configuration.html
mybatis.configuration.map-underscore-to-camel-case=true
mybatis.configuration.auto-mapping-unknown-column-behavior=warning
mybatis.configuration.use-generated-keys=true
使用原始的 Mybatis 有个问题就是,每个实体类的通用CURD操作等都需要自己写 xml 配置文件或者在对应的 Mapper 文件中写对应的 @Select
/@Insert
等注解来实现对应的功能。这是个特别耗时的重复工作。
解决这个问题有两种方式:
个人觉得集成通用 Mapper 相对简单,所以选择了该中方式。通用 Mapper 的作者本身也有写两种方式的对比。MyBatis 通用 Mapper3 文档
另外,为了方便使用支持物理分页,也需要集成分页插件 Mybatis_PageHelper。
<dependency>
<groupId>tk.mybatisgroupId>
<artifactId>mapper-spring-boot-starterartifactId>
<version>${mapper-spring-boot-starter.version}version>
dependency>
<dependency>
<groupId>com.github.pagehelpergroupId>
<artifactId>pagehelper-spring-boot-starterartifactId>
<version>${pagehelper-spring-boot-starter.version}version>
dependency>
编写基本的 CommonMapper 接口,继承通用 Mapper 的接口 Mapper
, MySqlMapper
, SelectByIdsMapper
, DeleteByIdsMapper
;其中有些接口只适用于特定的数据库,需要根据实际情况做调整。 其他的业务相关的 Mapper 则需要继承这个 CommonMapper
接口。
对应 service 层基本 Service 接口和实现则根据自身需要考虑是否需要。
import tk.mybatis.mapper.common.Mapper;
import tk.mybatis.mapper.common.MySqlMapper;
import tk.mybatis.mapper.common.ids.DeleteByIdsMapper;
import tk.mybatis.mapper.common.ids.SelectByIdsMapper;
/**
* 支持单表CURD和批量(MYSQL)操作的通用Mapper
* Created by Vio on 2017/11/6.
*/
public interface CommonMapper<T> extends Mapper<T>, MySqlMapper<T>, SelectByIdsMapper<T>, DeleteByIdsMapper<T> {
}
# 这里配置自己写的基本的CommonMapper
mapper.mappers=com.xx.xxx.app.mapper.CommonMapper
mapper.not-empty=false
mapper.identity=MYSQL
#pagehelper插件配置
pagehelper.helperDialect=mysql
pagehelper.reasonable=true
pagehelper.supportMethodsArguments=true
pagehelper.params=count=countSql
整合 Quartz 时最主要会遇到两个问题:
其实上面两个问题的根本都是因为 Job 的实例的创建和管理没有交给 Spring 来管理。下面给出可以解决上述两个问题的整合方式:
<dependency>
<groupId>org.springframework.bootgroupId>
<artifactId>spring-boot-starter-aopartifactId>
dependency>
<dependency>
<groupId>org.quartz-schedulergroupId>
<artifactId>quartzartifactId>
<version>${quartz.version}version>
dependency>
<dependency>
<groupId>org.quartz-schedulergroupId>
<artifactId>quartz-jobsartifactId>
<version>${quartz.version}version>
dependency>
JobFactoy
是 Quartz 提供的一个接口,其作用是用于创建 Job
实例,对应的在 Spring Quartz 中的实现类有 AdaptableJobFactory
和 SpringBeanJobFactory
;其中 AdaptableJobFactory
继承于 SpringBeanJobFactory
,我们看下其源码
public interface JobFactory {
Job newJob(TriggerFiredBundle bundle, Scheduler scheduler) throws SchedulerException;
}
public class AdaptableJobFactory implements JobFactory {
@Override
public Job newJob(TriggerFiredBundle bundle, Scheduler scheduler) throws SchedulerException {
try {
// 创建Job实例
Object jobObject = createJobInstance(bundle);
return adaptJob(jobObject);
}
catch (Exception ex) {
throw new SchedulerException("Job instantiation failed", ex);
}
}
// 这里可以看到,Job实例的创建都是通过getJobClass().newInstance()来创建的,并没有对应的代理类的创建
// 所以使用Spring AOP编程的时候所有需要被代理的Job任务实际上都不会有代理类生成,是无法使用Spring AOP编程的
protected Object createJobInstance(TriggerFiredBundle bundle) throws Exception {
return bundle.getJobDetail().getJobClass().newInstance();
}
protected Job adaptJob(Object jobObject) throws Exception {
if (jobObject instanceof Job) {
return (Job) jobObject;
}
else if (jobObject instanceof Runnable) {
return new DelegatingJob((Runnable) jobObject);
}
else {
throw new IllegalArgumentException("Unable to execute job class [" + jobObject.getClass().getName() +
"]: only [org.quartz.Job] and [java.lang.Runnable] supported.");
}
}
}
public class SpringBeanJobFactory extends AdaptableJobFactory implements SchedulerContextAware {
//此处省略部分代码 ...
@Override
protected Object createJobInstance(TriggerFiredBundle bundle) throws Exception {
// 可以看到SpringBeanJobFactory中是使用父类AdaptableJobFactory的方法来创建Job实例的,所以也不会有代理类的创建
Object job = super.createJobInstance(bundle);
if (isEligibleForPropertyPopulation(job)) {
BeanWrapper bw = PropertyAccessorFactory.forBeanPropertyAccess(job);
MutablePropertyValues pvs = new MutablePropertyValues();
if (this.schedulerContext != null) {
pvs.addPropertyValues(this.schedulerContext);
}
pvs.addPropertyValues(bundle.getJobDetail().getJobDataMap());
pvs.addPropertyValues(bundle.getTrigger().getJobDataMap());
if (this.ignoredUnknownProperties != null) {
for (String propName : this.ignoredUnknownProperties) {
if (pvs.contains(propName) && !bw.isWritableProperty(propName)) {
pvs.removePropertyValue(propName);
}
}
bw.setPropertyValues(pvs);
}
else {
bw.setPropertyValues(pvs, true);
}
}
return job;
}
//此处省略部分代码 ...
}
可以发现现有的 Spring 中对 JobFactory
的实现类都无法实现我们的 AOP 编程需求,所以就需要自定义一个 JobFactory
实现类了。如下
JobFactory
实现类/**
- 自定义JobFactory,将创建Job实例的操作交给Spring管理
- Created by Vio on 2017/11/8.
*/
public class MySpringBeanJobFactory extends AdaptableJobFactory {
@Autowired
private AutowireCapableBeanFactory beanFactory;
@Override
protected Object createJobInstance(TriggerFiredBundle bundle) throws Exception {
Object jobInstance;
Class<? extends Job> jobClass = bundle.getJobDetail().getJobClass();
// 这里将Job的实例创建喝管理交给Spring,使用Spring的beanFactory去获取Job实例,
// 获取不到的话就交由Spring的beanFactory自动创建一个,并根据名称自动注入和检查依赖关系
// 这样的话Job中就可以实现自动注入和实现AOP编程
try {
jobInstance = beanFactory.getBean(jobClass);
} catch (Exception e) {
jobInstance = beanFactory.createBean(jobClass, AutowireCapableBeanFactory.AUTOWIRE_BY_NAME, true);
}
return jobInstance;
}
}
/**
* Quartz配置类
* Created by Vio on 2017/11/2.
*/
@Configuration
public class QuartzConfiguration {
private static final String QUARTZ_CONFIG = "quartz.properties";
@Bean
public MySpringBeanJobFactory mySpringBeanJobFactory(){
return new MySpringBeanJobFactory();
}
@Bean
public SchedulerFactoryBean schedulerFactoryBean() {
SchedulerFactoryBean schedulerFactoryBean = new SchedulerFactoryBean();
// 配置使用自定义的JobFactory
schedulerFactoryBean.setJobFactory(mySpringBeanJobFactory());
schedulerFactoryBean.setAutoStartup(true);
// 设置quartz配置文件路径
schedulerFactoryBean.setConfigLocation(new ClassPathResource(QUARTZ_CONFIG));
return schedulerFactoryBean;
}
@Bean
public Scheduler scheduler() {
return schedulerFactoryBean().getScheduler();
}
}
动态任务的实现其实只需要从数据库中读取相关的数据,然后使用 Scheduler 的相关API重新设置对应的任务的调度即可。不过任务初始启动的时候需要将数据库中的任务取出来。加入到调度器中。( Spring 启动完成后执行某任务)实现如下:
/**
* 定时任务启动器:
* 应用启动时启动所有有效的定时任务
* Created by Vio on 2017/11/7.
*/
@Component
public class QuartzTaskStarter implements ApplicationListener<ContextRefreshedEvent> {
private static final Logger LOGGER = LoggerFactory.getLogger(QuartzTaskStarter.class);
@Autowired
private TaskSchedule taskSchedule;
@Autowired
private QuartzTaskService quartzTaskService;
@Override
public void onApplicationEvent(ContextRefreshedEvent event) {
try {
List<QuartzTaskEntity> tasks = quartzTaskService.selectAllValidTask();
for (QuartzTaskEntity task : tasks) {
taskSchedule.scheduleTask(task);
}
taskSchedule.startSchedule();
} catch (Exception e) {
LOGGER.error("Start all valid task fail.", e);
}
}
}
/**
* 定时任务调度器
* Created by Vio on 2017/11/2.
*/
@Component
public class TaskSchedule {
private static final Logger LOGGER = LoggerFactory.getLogger(TaskSchedule.class);
@Autowired
private Scheduler scheduler;
@Autowired
private QuartzTaskService quartzTaskService;
public ApiResult scheduleTask(QuartzTaskEntity task) {
boolean scheduled = false;
String msg = "Schedule Success!";
try {
Class<?> jobClass = Class.forName(task.getTaskClass());
if (Job.class.isAssignableFrom(jobClass)) {
JobDetail jobDetail = buildJobDetail(task, (Class<? extends Job>) jobClass);
Trigger trigger = buildTrigger(task);
scheduler.scheduleJob(jobDetail, trigger);
scheduled = true;
LOGGER.info(msg + "");
}
} catch (ClassNotFoundException e) {
msg = "Schedule Fail! Class not found!";
LOGGER.error(msg + "Task: " + task);
} catch (SchedulerException e) {
msg = "Schedule Fail! " + e.getMessage();
LOGGER.error(msg + "Task: " + task, e);
}
return ApiResult.build(scheduled, msg, task);
}
// 此处省略部分代码...
public ApiResult rescheduleTask(QuartzTaskEntity task) {
boolean flag = false;
String msg = "Reschedule task Success!";
try {
CronTrigger oldTrigger = (CronTrigger) scheduler.getTrigger(getTriggerKey(task));
if (!oldTrigger.getCronExpression().equalsIgnoreCase(task.getCron())) {
Trigger newTrigger = buildTrigger(task);
scheduler.rescheduleJob(getTriggerKey(task), newTrigger);
}
flag = true;
} catch (SchedulerException e) {
msg = "Reschedule task Fail! " + e.getMessage();
LOGGER.error(msg + "Task: " + task, e);
}
return ApiResult.build(flag, msg, task);
}
public synchronized void startSchedule() {
try {
if (scheduler.isShutdown()) {
scheduler.start();
}
} catch (SchedulerException e) {
LOGGER.error("Start scheduler fail.", e);
}
}
// 此处省略部分代码...
private JobDetail buildJobDetail(QuartzTaskEntity task, Class<? extends Job> clazz) {
return JobBuilder.newJob(clazz).withIdentity(getTaskName(task), getTaskGroup(task)).build();
}
private Trigger buildTrigger(QuartzTaskEntity task) {
return TriggerBuilder.newTrigger().withIdentity(getTaskName(task), getTaskGroup(task)).startNow().withSchedule(CronScheduleBuilder.cronSchedule(task.getCron())).build();
}
private JobKey getJobKey(QuartzTaskEntity task) {
return JobKey.jobKey(getTaskName(task), getTaskGroup(task));
}
private TriggerKey getTriggerKey(QuartzTaskEntity task) {
return TriggerKey.triggerKey(getTaskName(task), getTaskGroup(task));
}
private String getTaskName(QuartzTaskEntity task) {
return StringUtils.isEmpty(task.getTaskName()) ? task.getTaskClass() : task.getTaskName();
}
private String getTaskGroup(QuartzTaskEntity task) {
return StringUtils.isEmpty(task.getTaskGroup()) ? Scheduler.DEFAULT_GROUP : task.getTaskGroup();
}
}
# 简单实用内存类存储任务状态
# 更多配置使用请访问Quartz官网http://www.quartz-scheduler.org/documentation/quartz-2.2.x/configuration/ConfigMain.html
org.quartz.scheduler.instanceName = MyScheduler
org.quartz.threadPool.threadCount = 3
org.quartz.jobStore.class = org.quartz.simpl.RAMJobStore
现在 Druid 官方已经给出了一个 starter 的依赖。整合 Druid 已经非常简单了,只需要添加对应的依赖和配置即可
<dependency>
<groupId>mysqlgroupId>
<artifactId>mysql-connector-javaartifactId>
<scope>runtimescope>
dependency>
<dependency>
<groupId>com.alibabagroupId>
<artifactId>druid-spring-boot-starterartifactId>
<version>${druid-spring-boot-starter.version}version>
dependency>
spring.datasource.type=com.alibaba.druid.pool.DruidDataSource
# druid
# see more config about druid: https://github.com/alibaba/druid/tree/master/druid-spring-boot-starter
spring.datasource.druid.initial-size=5
spring.datasource.druid.max-active=20
spring.datasource.druid.min-idle=5
spring.datasource.druid.pool-prepared-statements=true
spring.datasource.druid.max-pool-prepared-statement-per-connection-size=20
spring.datasource.druid.max-wait=60000
spring.datasource.druid.time-between-eviction-runs-millis=60000
spring.datasource.druid.min-evictable-idle-time-millis=300000
spring.datasource.druid.validation-query=SELECT 1 FROM DUAL
spring.datasource.druid.test-on-borrow=false
spring.datasource.druid.test-on-return=false
spring.datasource.druid.test-while-idle=true
spring.datasource.druid.filters= stat,wall,slf4j
spring.datasource.druid.filter.stat.slow-sql-millis= 5000
在编写 Dao 的测试用例的时候,会对数据库中的数据进行操作。但是一般我们不想测试用例的运行对我们的开发库/测试服务库中的数据造成污染。那么可以考虑集成 H2 数据库来运行测试用例。集成如下:
<dependency>
<groupId>com.h2databasegroupId>
<artifactId>h2artifactId>
<scope>testscope>
dependency>
<dependency>
<groupId>org.springframework.bootgroupId>
<artifactId>spring-boot-starter-testartifactId>
<scope>testscope>
dependency>
此处的 application.properties 文件是 test 目录下的配置文件。
# 数据源连接
spring.datasource.driver-class-name=org.h2.Driver
spring.datasource.url=jdbc:h2:mem:test
spring.datasource.username=root
spring.datasource.password=
# 设置测试启动时执行的创建schema的脚本文件
spring.datasource.schema=classpath:db/schema.sql
# 设置测试启动时执行的插入数据的脚本文件
spring.datasource.data=classpath:db/data.sql
sql 脚本文件需要自己创建并写入 sql 语句。其余的测试用例的编写和 Spring 官方说明一致。
/**
* 定时任务实体服务测试类
* Created by Vio on 2017/11/7.
*/
@RunWith(SpringRunner.class)
@SpringBootTest
public class QuartzServiceTest {
@Autowired
QuartzTaskService quartzTaskService;
@Test
public void testTaskExist() {
QuartzTaskEntity quartzTaskEntity = new QuartzTaskEntity();
quartzTaskEntity.setTaskClass("TestTask2");
quartzTaskEntity.setState(0);
int inserted = quartzTaskService.insert(quartzTaskEntity);
Assert.assertEquals(1, inserted);
Assert.assertEquals(true, quartzTaskService.taskExist(quartzTaskEntity));
}
}
Done。大功告成。