概述
在做数据产品或平台系统时,经常会遇到类似如下截图中,有截至日期的定时调度任务的需求。即定时任务只在指定的开始日期-截至日期里指定的时间里执行。具体的业务需求场景,如营销活动的看板数据的订阅邮件,推送名单的活动,活动(双十一,六一八)结束后就不再需要执行任务。
注:因为使用的Java语言的定时调度平台是XXL-JOB。
需求分析
查看XXL-JOB管理平台:
发现没有任务开始日期和结束日期的概念。这一点可以在源码com.xxl.job.core.biz.model.JobEntity.java
得到验证:
public class JobEntity implements Serializable {
private int id;
private int jobGroup;
private String jobCron;
private String jobDesc;
private Date addTime;
private Date updateTime;
private String author;
private String alarmEmail;
private String executorRouteStrategy;
private String executorHandler;
private String executorParam;
private String executorBlockStrategy;
private String executorFailStrategy;
private String glueType;
private String glueSource;
private String glueRemark;
private Date glueUpdatetime;
private String childJobKey;
private String jobStatus;
}
注:GitHub的代码,找不到这个类,奇怪。
实现
针对截图1的表设计如下(省略无关字段):
create table job (
job_id bigint auto_increment primary key,
job_name varchar(200) null comment '订阅名称',
cron_exp varchar(200) null comment 'cron',
cron_exp_status tinyint(1) default 1 null comment '1表示任务有效,0表示任务关闭',
start_date timestamp null comment '订阅开始时间 yyyy-MM-dd',
end_date timestamp null comment '订阅结束时间 yyyy-MM-dd',
xxl_job_id varchar(50) null comment 'xxl-job id',
is_active tinyint(1) default 1 null comment '逻辑删除,1表示有效'
);
新增一个JobHandler,每天0点2分执行,判断当前日期(天粒度)是否达到end_date
,如果达到,则设置cron_exp_status=0
,并同步到XXL-JOB平台,XXL-JOB平台则不再继续调度此任务。
/**
* 订阅邮件job有效期结束后,暂停任务
*
* @author johnny
*/
@Slf4j
@Component
@JobHander(value = "stopExecuteJobHandler")
public class StopExecuteJobHandler extends IJobHandler {
@Resource
private JobMapper jobMapper;
@Resource
private JobService jobService;
@Override
public ReturnT<String> execute(String... strings) throws Exception {
List<DashboardJob> jobList = jobMapper.getAllStoppedJob();
if (CollectionUtils.isEmpty(jobList)) {
log.info(String.format("%s: 没有需要暂停调度的任务", DateUtil.getNow()));
return ReturnT.SUCCESS;
}
for (DashboardJob item : jobList) {
jobMapper.stopJob(item.getId());
jobService.disableJob(item.getXxlJobId());
}
return ReturnT.SUCCESS;
}
}
对应的mapper.xml
文件:
<select id="getAllStoppedJob" resultType="com.xy.johnny.po.DashboardJob">
SELECT job_id AS id,
xxljob_id AS xxlJobId
FROM job WHERE end_date < now() AND isactive = 1 AND cron_exp_status = 1
</select>
<update id="stopJob">
UPDATE job SET cron_exp_status = 0 WHERE job_id = #{jobId} AND isactive = 1
</update>
封装XXL-JOB提供的JobRestApi的JobService.java
:
@Resource
private JobRestApi jobRestApi;
/**
* 暂停Job的调度
*/
public void disableJob(Integer id) throws Exception {
if (id == null) {
return;
}
ReturnT<String> returnT = jobRestApi.disableJob(id);
if (returnT.getCode() != CODE) {
throw new Exception(returnT.getMsg());
}
}
反思
Quartz任务有任务截至有效期,不明白为啥XXL-JOB调度平台没有这个概念。
看GitHub动态,XXL-JOB几乎处于停止维护状态,需要forkGit仓库后二次开发。
参考
分布式任务调度平台XXL-JOB深度实战