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;
    }

}
相关推荐
武子康5 分钟前
Java-168 Neo4j CQL 实战:WHERE、DELETE/DETACH、SET、排序与分页
java·开发语言·数据库·python·sql·nosql·neo4j
Filotimo_5 分钟前
SpringBoot3入门
java·spring boot·后端
通往曙光的路上17 分钟前
SpringIOC-注解
java·开发语言
一 乐27 分钟前
校园墙|校园社区|基于Java+vue的校园墙小程序系统(源码+数据库+文档)
java·前端·数据库·vue.js·spring boot·后端·小程序
TT哇27 分钟前
【面经 每日一题】面试题16.25.LRU缓存(medium)
java·算法·缓存·面试
青云交29 分钟前
Java 大视界 -- 基于 Java 的大数据联邦学习在跨行业数据协同创新中的实践突破
java·分布式计算·隐私保护·apache flink·大数据联邦学习·跨行业数据协同·安全通信
合作小小程序员小小店29 分钟前
桌面开发,在线%考试管理%系统,基于eclipse,java,swing,mysql数据库。
java·数据库·mysql·eclipse·jdk
oioihoii33 分钟前
C/C++混合项目中的头文件管理:.h与.hpp的分工与协作
java·c语言·c++
一瓢一瓢的饮 alanchan1 小时前
Flink原理与实战(java版)#第2章 Flink的入门(第二节Flink简介)
java·大数据·flink·kafka·实时计算·离线计算·流批一体化计算
golang学习记1 小时前
🍵 Go Queryx 入门指南:让数据库操作像喝奶茶一样丝滑!
后端