博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
springboot开发之定时器quartz 定时任务调度(压缩版,抽取quartz的单个任务表实现)...
阅读量:6871 次
发布时间:2019-06-26

本文共 10729 字,大约阅读时间需要 35 分钟。

hot3.png

前言

老了, 记不住了, 好记性不如烂笔头;

没想到曾经过目不忘的我, 也有这么一天, 岁月蹉跎,学习一天不如一天

难受

Quartz可以用来做什么?

Quartz是一个任务调度框架。比如你遇到这样的问题

  • 想每月25号,信用卡自动还款
  • 想每年4月1日自己给当年暗恋女神发一封匿名贺卡
  • 想每隔1小时,备份一下自己的爱情动作片 学习笔记到云盘

这些问题总结起来就是:在某一个有规律的时间点干某件事。并且时间的触发的条件可以非常复杂(比如每月最后一个工作日的17:50),复杂到需要一个专门的框架来干这个事。 Quartz就是来干这样的事,你给它一个触发条件的定义,它负责到了时间点,触发相应的Job起来干活。

汇总以上的, 就是两点: 1.定时定点的执行任务(一次性), 2.循环执行任务(循环);

一个简单的示例

import static org.quartz.DateBuilder.newDate;import static org.quartz.JobBuilder.newJob;import static org.quartz.SimpleScheduleBuilder.simpleSchedule;import static org.quartz.TriggerBuilder.newTrigger;import java.util.GregorianCalendar;import org.quartz.JobDetail;import org.quartz.Scheduler;import org.quartz.Trigger;import org.quartz.impl.StdSchedulerFactory;import org.quartz.impl.calendar.AnnualCalendar;public class QuartzTest {    public static void main(String[] args) {        try {            //创建scheduler            Scheduler scheduler = StdSchedulerFactory.getDefaultScheduler();            //定义一个Trigger            Trigger trigger = newTrigger().withIdentity("trigger1", "group1") //定义name/group                .startNow()//一旦加入scheduler,立即生效                .withSchedule(CronScheduleBuilder.cronSchedule("0 0/2 8-17 * * ?"))//每天8:00-17:00,每隔2分钟执行一次                .build();            //定义一个JobDetail            JobDetail job = newJob(HelloQuartz.class) //定义Job类为HelloQuartz类,这是真正的执行逻辑所在                .withIdentity("job1", "group1") //定义name/group                .usingJobData("name", "quartz") //定义属性                .build();            //加入这个调度            scheduler.scheduleJob(job, trigger);            //启动之            scheduler.start();            //运行一段时间后关闭            Thread.sleep(10000);            scheduler.shutdown(true);        } catch (Exception e) {            e.printStackTrace();        }    }}
import java.util.Date;import org.quartz.DisallowConcurrentExecution;import org.quartz.Job;import org.quartz.JobDetail;import org.quartz.JobExecutionContext;import org.quartz.JobExecutionException;/** 具体业务类 **/public class HelloQuartz implements Job {    public void execute(JobExecutionContext context) throws JobExecutionException {        JobDetail detail = context.getJobDetail();        String name = detail.getJobDataMap().getString("name");        System.out.println("say hello to " + name + " at " + new Date());    }}

针对以上代码做个小白可能会一脸懵逼;这里做个解释

  • Scheduler:调度器。所有的调度都是由它控制。
  • Trigger: 定义触发的条件。例子中,它的类型是SimpleTrigger,每隔1秒中执行一次(什么是SimpleTrigger下面会有详述)。
  • JobDetail & Job: JobDetail 定义的是任务数据,而真正的执行逻辑是在Job中,例子中是HelloQuartz。 为什么设计成JobDetail + Job,不直接使用Job?这是因为任务是有可能并发执行,如果Scheduler直接使用Job,就会存在对同一个Job实例并发访问的问题。而JobDetail & Job 方式,sheduler每次执行,都会根据JobDetail创建一个新的Job实例,这样就可以规避并发访问的问题。

讲重点

CronTrigger

适合于更复杂的任务,它支持类型于Linux Cron的语法(并且更强大)。基本上它覆盖了以上三个Trigger的绝大部分能力(但不是全部)—— 当然,也更难理解。

它适合的任务类似于:每天0:00,9:00,18:00各执行一次。

它的属性只有:

  • Cron表达式
位置 时间域 允许值 特殊值
1 0-59 , - * /
2 分钟 0-59 , - * /
3 小时 0-23 , - * /
4 日期 1-31 , - * ? / L W C
5 月份 1-12 , - * /
6 星期 1-7 , - * ? / L C #
7 年份(可选) 1-31 , - * /

星号():可用在所有字段中,表示对应时间域的每一个时刻,例如, 在分钟字段时,表示“每分钟”;

问号(?):该字符只在日期和星期字段中使用,它通常指定为“无意义的值”,相当于点位符;

减号(-):表达一个范围,如在小时字段中使用“10-12”,则表示从10到12点,即10,11,12;

逗号(,):表达一个列表值,如在星期字段中使用“MON,WED,FRI”,则表示星期一,星期三和星期五;

斜杠(/):x/y表达一个等步长序列,x为起始值,y为增量步长值。如在分钟字段中使用0/15,则表示为0,15,30和45秒,而5/15在分钟字段中表示5,20,35,50,你也可以使用*/y,它等同于0/y;

L:该字符只在日期和星期字段中使用,代表“Last”的意思,但它在两个字段中意思不同。L在日期字段中,表示这个月份的最后一天,如一月的31号,非闰年二月的28号;如果L用在星期中,则表示星期六,等同于7。但是,如果L出现在星期字段里,而且在前面有一个数值X,则表示“这个月的最后X天”,例如,6L表示该月的最后星期五;

W:该字符只能出现在日期字段里,是对前导日期的修饰,表示离该日期最近的工作日。例如15W表示离该月15号最近的工作日,如果该月15号是星期六,则匹配14号星期五;如果15日是星期日,则匹配16号星期一;如果15号是星期二,那结果就是15号星期二。但必须注意关联的匹配日期不能够跨月,如你指定1W,如果1号是星期六,结果匹配的是3号星期一,而非上个月最后的那天。W字符串只能指定单一日期,而不能指定日期范围;

LW组合:在日期字段可以组合使用LW,它的意思是当月的最后一个工作日;

井号(#):该字符只能在星期字段中使用,表示当月某个工作日。如6#3表示当月的第三个星期五(6表示星期五,#3表示当前的第三个),而4#5表示当月的第五个星期三,假设当月没有第五个星期三,忽略不触发;

C:该字符只在日期和星期字段中使用,代表“Calendar”的意思。它的意思是计划所关联的日期,如果日期没有被关联,则相当于日历中所有日期。例如5C在日期字段中就相当于日历5日以后的第一天。1C在星期字段中相当于星期日后的第一天。

Cron表达式对特殊字符的大小写不敏感,对代表星期的缩写英文大小写也不敏感。

一些例子:

表示式 说明
0 0 12 * * ? 每天12点运行
0 15 10 ? * * 每天10:15运行
0 15 10 * * ? 每天10:15运行
0 15 10 * * ? * 每天10:15运行
0 15 10 * * ? 2008 在2008年的每天10:15运行
0 * 14 * * ? 每天14点到15点之间每分钟运行一次,开始于14:00,结束于14:59。
0 0/5 14 * * ? 每天14点到15点每5分钟运行一次,开始于14:00,结束于14:55。
0 0/5 14,18 * * ? 每天14点到15点每5分钟运行一次,此外每天18点到19点每5钟也运行一次。
0 0-5 14 * * ? 每天14:00点到14:05,每分钟运行一次。
0 10,44 14 ? 3 WED 3月每周三的14:10分到14: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 2007-2009 在2007,2008,2009年每个月的最后一个星期五的10:15分运行。
0 15 10 ? * 6#3 每月第三个星期五的10:15分运行。

 

JobDetail & Job

要定义一个任务,需要干几件事

  1. 创建一个org.quartz.Job的实现类,并实现实现自己的业务逻辑。比如上面的DoNothingJob。
  2. 定义一个JobDetail,引用这个实现类
  3. 加入scheduleJob

Quartz调度一次任务,会干如下的事:

  1. JobClass jobClass=JobDetail.getJobClass()
  2. Job jobInstance=jobClass.newInstance()。所以Job实现类,必须有一个public的无参构建方法。
  3. jobInstance.execute(JobExecutionContext context)。JobExecutionContext是Job运行的上下文,可以获得Trigger、Scheduler、JobDetail的信息。

也就是说,每次调度都会创建一个新的Job实例,这样的好处是有些任务并发执行的时候,不存在对临界资源的访问问题——当然,如果需要共享JobDataMap的时候,还是存在临界资源的并发访问的问题。

JobDataMap

ob都次都是newInstance的实例,那我怎么传值给它? 比如我现在有两个发送邮件的任务,一个是发给"liLei",一个发给"hanmeimei",不能说我要写两个Job实现类LiLeiSendEmailJob和HanMeiMeiSendEmailJob。实现的办法是通过JobDataMap。

每一个JobDetail都会有一个JobDataMap。JobDataMap本质就是一个Map的扩展类,只是提供了一些更便捷的方法,比如getString()之类的。

我们可以在定义JobDetail,加入属性值,方式有二:

newJob().usingJobData("age", 18) //加入属性到ageJobDataMap or job.getJobDataMap().put("name", "quertz"); //加入属性name到JobDataMap

然后在Job中可以获取这个JobDataMap的值,方式同样有二:

public class HelloQuartz implements Job {    private String name;    public void execute(JobExecutionContext context) throws JobExecutionException {        JobDetail detail = context.getJobDetail();        JobDataMap map = detail.getJobDataMap(); //方法一:获得JobDataMap        System.out.println("say hello to " + name + "[" + map.getInt("age") + "]" + " at "                           + new Date());    }    //方法二:属性的setter方法,会将JobDataMap的属性自动注入    public void setName(String name) {         this.name = name;    }}

对于同一个JobDetail实例,执行的多个Job实例,是共享同样的JobDataMap,也就是说,如果你在任务里修改了里面的值,会对其他Job实例(并发的或者后续的)造成影响。

除了JobDetail,Trigger同样有一个JobDataMap,共享范围是所有使用这个Trigger的Job实例。

Job并发

Job是有可能并发执行的,比如一个任务要执行10秒中,而调度算法是每秒中触发1次,那么就有可能多个任务被并发执行。

有时候我们并不想任务并发执行,比如这个任务要去”获得数据库中所有未发送邮件的名单“,如果是并发执行,就需要一个数据库锁去避免一个数据被多次处理。这个时候一个@DisallowConcurrentExecution解决这个问题。

就是这样

public class DoNothingJob implements Job {    @DisallowConcurrentExecution    public void execute(JobExecutionContext context) throws JobExecutionException {        System.out.println("do nothing");    }}

注意,@DisallowConcurrentExecution是对JobDetail实例生效,也就是如果你定义两个JobDetail,引用同一个Job类,是可以并发执行的。

JobExecutionException

Job.execute()方法是不允许抛出除JobExecutionException之外的所有异常的(包括RuntimeException),所以编码的时候,最好是try-catch住所有的Throwable,小心处理。

文中内容也是基本来自本篇: http://www.cnblogs.com/drift-ice/p/3817269.html 

 

重点来了, 如何集成到springboot 呢

集成至springboot

1. 拆分表设计

不重复早轮子了, 刚好发现它, 很有意思;

https://www.cnblogs.com/softidea/p/7444998.html

原理就是: 单独拆分 定时表出来, 查询任务, 添加任务, 修改任务状态, 等均在这个表;

具体实现,完全可以参照上面说的

在springboot初始化时, 

新增config类

package com.xx.xx.xxx.config;import com.xxx.xxx.xxx.bean.dto.task.TaskConstant;import com.xxx.xxx.xxx.bean.entity.XXX;import com.xxx.xxx.xxx.env.quartz.JobTest;import com.xxx.xxx.xxx.env.quartz.MyJobFactory;import com.xxx.xxx.xxx.service.TaskService;import lombok.extern.slf4j.Slf4j;import org.quartz.*;import org.quartz.impl.StdSchedulerFactory;import org.springframework.beans.factory.annotation.Autowired;import org.springframework.scheduling.quartz.SchedulerFactoryBean;import org.springframework.stereotype.Component;import org.springframework.stereotype.Service;import javax.annotation.PostConstruct;import java.util.List;import static org.quartz.JobBuilder.newJob;import static org.quartz.SimpleScheduleBuilder.simpleSchedule;import static org.quartz.TriggerBuilder.newTrigger;/** * 启动项目时定时调用任务 */@Service@Slf4jpublic class QuartzConfig {    @Autowired    TaskService taskService;    @Autowired    Scheduler myScheduler;    @PostConstruct  // spirngIOC初始化后, 标识执行该方法;    private void startQuartzConfig() {        log.info("启动初始化完成,开始启动定时任务");        Scheduler sched = null;        try {            sched = myScheduler;            sched.start();            List
xxxList= taskService.getALLTasks(); // 查询数据中, 所有已添加好的任务表 for (XXX t : xxxList) { JobDetail job = newJob(JobTest.class) .usingJobData(TaskConstant.SERVICE_ID, t.getId()) .withIdentity(t.getId() + ":task", TaskConstant.GROUP_NAME) .build(); // Trigger the job to run now, and then every 40 seconds Trigger trigger = newTrigger() .withIdentity(kz01.getId() + ":trigger", TaskConstant.GROUP_NAME) .startNow() .withSchedule(CronScheduleBuilder.cronSchedule("*/5 * * * * ?") .build(); sched.scheduleJob(job, trigger); } } catch (SchedulerException e) { log.error("定时任务执行时出错!", e); } log.info("定时任务初始化结束"); }}
package com.xxx.xxx.xxx.env.quartz;import com.alibaba.fastjson.JSONObject;import com.xxx.xxx.xxx.bean.dto.ResultSupport;import com.xxx.xxx.xxx.bean.dto.auditcase.CasePointCountDTO;import com.xxx.xxx.xxx.bean.dto.task.GroupingCaseDTO;import com.xxx.xxx.xxx.bean.dto.task.TaskConstant; import com.xxx.xxx.xxx.controller.taskApi.TaskController;import com.xxx.xxx.xxx.service.TaskService;import lombok.extern.slf4j.Slf4j;import org.quartz.DisallowConcurrentExecution;import org.quartz.Job;import org.quartz.JobExecutionContext;import org.quartz.JobExecutionException;import org.springframework.beans.BeansException;import org.springframework.beans.factory.annotation.Autowired;import org.springframework.context.ApplicationContext;import org.springframework.context.ApplicationContextAware;import javax.xml.bind.util.JAXBSource;import java.util.Date;/** job启动,可有效避免同步job执行*/@DisallowConcurrentExecution@Slf4jpublic class JobTest implements Job {    @Autowired    TaskService taskService;    @Autowired    TaskController taskController; // 定时执行方法    @Override    public void execute(JobExecutionContext jobExecutionContext) throws JobExecutionException {        Long serviceId = (Long) jobExecutionContext.getJobDetail().getJobDataMap().get(TaskConstant.SERVICE_ID); //获取传参值        log.info("开始调用定时任务" + serviceId);        Tast tempt = taskService.getTask(serviceId);//获取单个任务详情                doTask();    }    public static  void doTask() {         // 业务逻辑    }}

 

 

总结:

quartz 有多种分配方式, 重的, 有创建5个表, 可具体跟踪定位任务执行情况,适合大型项目;

比如这篇文章: https://www.cnblogs.com/nick-huang/p/8456272.html 

拆分个人任务表(job_task)集成到项目中, 如上 , 精简版, 仅仅用来一个小量的定时业务;

 

完结撒花;

 

 

 

 

 

 

 

 

 

 

 

 

 

 

转载于:https://my.oschina.net/java1314/blog/3024222

你可能感兴趣的文章
view属性大全
查看>>
Java文件编码示例
查看>>
CactiFans V1.0中文版发布
查看>>
HTML如何显示小于号“<”等特殊符号?
查看>>
别伤了虚拟桌面管理员的"心"
查看>>
Windows系统使用IntelliJ IDEA 搭建Hadoop的开发调试环境(一)
查看>>
yum安装lamp
查看>>
Web.Config文件中数据库连接配置
查看>>
[Unity 3D] Unity 3D 性能优化 (一)
查看>>
spring Quartz定时任务调度 时间设置
查看>>
SymmetricDS: 数据库数据同步Database synchronization
查看>>
Disabling OOM killer on Ubuntu 14.04
查看>>
VBS备份脚本
查看>>
CentOS 6.5 自动安装镜像
查看>>
Storm与Spark Streaming比较
查看>>
我的友情链接
查看>>
Exchange Server 运维管理01:Exchange中Active Directory 有什么用?
查看>>
dhcp服务在企业中的应用
查看>>
linux系统管理之四:服务状态
查看>>
VMware View FAQ[一]
查看>>