SpringBoot 定时任务+Quartz

前言:

1、定时任务技术:

  • JDK 的 Timer, 定义多个定时任务,其中某个任务出现异常,当时整个定时任务终止。
  • Spring Task , 不支持 持久化与分布式部署,所有任务是单线程执行的
  • Quartz,支持持久化多线程执行。

2、Spring Boot 整合Quartz

  • Job: 任务。定时执行的具体任务内容
  • JobDetail: 任务详情。即与任务相关的其他配置信息
  • Trigger: 触发器,主要负责描述任务执行的时间规则
  • Scheduler: 调度器。
  • 将Job和Trigger整合起来之间的关系: Job:JobDetail->1:N JobDetail:Trigger->1:N Trigger:JobDetail->1:1

3、引入pom文件:

javascript 复制代码
 <dependency>
    <groupId>org.springframework.boot</groupId>
    <artifactId>spring-boot-starter-quartz</artifactId>
 </dependency>

4、创建执行任务,Job (自己的定时任务)

对应代码:

javascript 复制代码
@Slf4j
@DisallowConcurrentExecution
@PersistJobDataAfterExecution
public class MyJob extends QuartzJobBean {
    @Override
    protected void executeInternal(JobExecutionContext context) throws JobExecutionException {
        System.out.println("任务开始执行");
        JobDetail jobDetail = context.getJobDetail();
        System.out.println("名字" + jobDetail.getKey().getName());
        System.out.println("类名--->" + jobDetail.getJobClass().getName());
        System.out.println("本次执行的时间为---》" + context.getFireTime());
        System.out.println("下次执行的时间为---》" + context.getNextFireTime());
        System.out.println("任务执行完毕");
        System.out.println("============================");
    }
} 

5、创建JobDtail 和 Trigger


在执行任务的类上添加注解。

  • PersistJobDataAfterExecution ,有状态的,在定时任务执行的Job中获取的JobExecutionContext 是同一个。比如在JobDetail 中,usingJobData("count",1);进行共享数据的初始化,如果不使用这个注解在执行任务上,会获取不同的JobExecutionContext 。
  • DisallowConcurrentExecution 在时间间隔内,如果本任务没有执行完成,下一个任务不会启动,一直到本任务结束。下一个任务才开始执行。

创建配置类:


解释:

持久化 .storeDurably() 。如果没有持久化,没有对应的触发器,失去执行的条件,存在没有意义,就会将其删除。

如果有持久化,没有对应的触发器,还是会保留 JobDetail。

对应代码:

javascript 复制代码
@Configuration
public class QuartzConfiguration {

    //JobDetail
     @Bean(value = "jobDetail")
    public JobDetail jobDetail(){
    return JobBuilder.newJob(MyJob.class)
            .storeDurably()
            //唯一标识
            .withIdentity("jobDetail")
            .build();
    }
    @Bean
    // @Qualifier 引用指定的bean 的实例对象。
    public Trigger trigger(@Qualifier("jobDetail")JobDetail jobDetail){
        Trigger jobDetail1 = TriggerBuilder.newTrigger()
                .forJob(jobDetail)
                //唯一标识和上面的JobDetail是配对的。
                .withIdentity("jobDetail")
                //这个表达式可以写到yml 中进行引用,这样以后更改直接改配置文件即可。
                //每两秒钟执行一次
                .withSchedule(CronScheduleBuilder.cronSchedule("0/2 * * * * ?"))
                .build();
        return jobDetail1;
    }

}

6、cron表达式

java 复制代码
tips:
  一般使用 六位,最后一位年 一般不使用。

整体代码实例演示:
pom文件

javascript 复制代码
    <dependency>
        <groupId>org.springframework.boot</groupId>
        <artifactId>spring-boot-starter-quartz</artifactId>
    </dependency>

1、自己的定时任务Job

javascript 复制代码
@Slf4j
@DisallowConcurrentExecution
@PersistJobDataAfterExecution
public class MyJob extends QuartzJobBean {
    @Override
    protected void executeInternal(JobExecutionContext context) throws JobExecutionException {
        System.out.println("任务开始执行");
        JobDetail jobDetail = context.getJobDetail();
        System.out.println("名字" + jobDetail.getKey().getName());
        System.out.println("类名--->" + jobDetail.getJobClass().getName());
        System.out.println("本次执行的时间为---》" + context.getFireTime());
        System.out.println("下次执行的时间为---》" + context.getNextFireTime());
        System.out.println("任务执行完毕");
        System.out.println("============================");
    }
} 

2、定时任务的配置类

javascript 复制代码
@Configuration
public class QuartzConfiguration {

    //JobDetail
     @Bean(value = "jobDetail")
    public JobDetail jobDetail(){
    return JobBuilder.newJob(MyJob.class)
            .storeDurably()
            //唯一标识
            .withIdentity("jobDetail")
            .build();
    }
    @Bean
    //Trigger
    public Trigger trigger(@Qualifier("jobDetail")JobDetail jobDetail){
        Trigger jobDetail1 = TriggerBuilder.newTrigger()
                .forJob(jobDetail)
                //唯一标识和上面的JobDetail是配对的。
                .withIdentity("jobDetail")
                //这个表达式可以写到yml 中进行引用,这样以后更改直接改配置文件即可。
                //每两秒钟执行一次
                .withSchedule(CronScheduleBuilder.cronSchedule("0/2 * * * * ?"))
                .build();
        return jobDetail1;
    }

}
相关推荐
禁默25 分钟前
深入浅出:AWT的基本组件及其应用
java·开发语言·界面编程
Cachel wood31 分钟前
python round四舍五入和decimal库精确四舍五入
java·linux·前端·数据库·vue.js·python·前端框架
Code哈哈笑34 分钟前
【Java 学习】深度剖析Java多态:从向上转型到向下转型,解锁动态绑定的奥秘,让代码更优雅灵活
java·开发语言·学习
gb421528737 分钟前
springboot中Jackson库和jsonpath库的区别和联系。
java·spring boot·后端
程序猿进阶37 分钟前
深入解析 Spring WebFlux:原理与应用
java·开发语言·后端·spring·面试·架构·springboot
zfoo-framework1 小时前
【jenkins插件】
java
风_流沙1 小时前
java 对ElasticSearch数据库操作封装工具类(对你是否适用嘞)
java·数据库·elasticsearch
颜淡慕潇1 小时前
【K8S问题系列 |19 】如何解决 Pod 无法挂载 PVC问题
后端·云原生·容器·kubernetes
ProtonBase1 小时前
如何从 0 到 1 ,打造全新一代分布式数据架构
java·网络·数据库·数据仓库·分布式·云原生·架构