SpringBoot - 集成Quartz框架之常见问题(五)

写在前面

SpringBoot - 集成Quartz框架之CRON表达式
SpringBoot - 集成Quartz框架之Quartz简介(一)
SpringBoot - 集成Quartz框架之常用配置(二)
SpringBoot - 集成Quartz框架之具体步骤(三)
SpringBoot - 集成Quartz框架之独立数据源(四)
SpringBoot - 集成Quartz框架之常见问题(五)
SpringBoot - 集成Quartz框架之@DisallowConcurrentExecution注解详解(六)

1. 什么是失火策略?

场景描述

quartz-scheduler-misfire
失火策略也可以叫错误触发策略,由于JOB到了触发时间时没有被触发执行,而且被执行的延迟时间也超过了Quartz配置的misfireThreshold阈值,此时就会触发失火策略。有哪些原因没有被触发执行呢?
(1)JOB达到触发时间点时,所有线程都被其他JOB占用,没有可用线程;
(2)JOB达到触发时间点时,调度器停止了;
(3)JOB使用了@DisallowConcurrentExecution注解,使得JOB不能并发执行,当达到下一个JOB达到触发时间点时,上一个还未执行完毕;
(4)JOB设置的开始时间为已经过去的时间。

配置说明

针对CronTrigger触发器,定义了三种失火策略,如下:

.withMisfireHandlingInstructionIgnoreMisfires()
MISFIRE_INSTRUCTION_IGNORE_MISFIRE_POLICY = -1
所有未触发的执行都会立即执行,然后触发器再按计划运行。

.withMisfireHandlingInstructionFireAndProceed()
MISFIRE_INSTRUCTION_FIRE_ONCE_NOW = 1
立即执行第一个错误的执行并丢弃其他(即所有错误的执行合并在一起),也就是说无论错过了多少次触发器的执行,都只会立即执行一次,然后触发器再按计划运行。

.withMisfireHandlingInstructionDoNothing()
MISFIRE_INSTRUCTION_DO_NOTHING = 2
所有未触发的执行都将被丢弃,然后再触发器的下一个调度周期按计划运行。

2. 什么是JobStore?

JobStore用于跟踪提供给调度器的所有的数据,如:JOBS,TRIGGERS,日历等等。分为RAMJobStore和JDBCJobStore。
(1)RAMJobStore,是使用最简单的JobStore,也是性能最高的(在CPU时间方面)。缺点是当应用程序结束/崩溃时,所有调度信息都将丢失,这意味着RAMJobStore无法履行JOBS和TRIGGERS上的“非易失性”设置。
(2)JDBCJobStore,通过JDBC将其所有数据保存在数据库中。已被广泛应用于Oracle,PostgreSQL,MySQL,MS SQLServer,HSQLDB和DB2。要使用JDBCJobStore,首先要创建一组数据库表以供Quartz使用。可以在Quartz发行版的“docs/dbTables”目录中找到SQL脚本。

3. 触发器的状态有哪些?

状态描述
WAITING创建任务后,触发器的默认状态
ACQUIRED当到达触发时间时的状态
EXECUTING运行中
COMPLETE完成(任务结束)
BLOCKED阻塞状态
ERROR错误状态
PAUSED暂停状态
PAUSED_BLOCKED暂停且阻塞状态(非并发下)
DELETED删除状态

4. 如何获取SQL脚本?

官网:http://www.quartz-scheduler.org/
下载:quartz-2.4.0-SNAPSHOT-distribution.tar.gz
切换:quartz-2.4.0-SNAPSHOT-distribution/quartz-2.4.0-SNAPSHOT/src/org/quartz/impl/jdbcjobstore/路径下查找对应的数据库脚本。
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述

5. @QuartzDataSource注解的作用?

To have Quartz use a DataSource other than the application’s main DataSource, declare a DataSource bean, annotating its @Bean method with @QuartzDataSource. Doing so ensures that the Quartz-specific DataSource is used by both the SchedulerFactoryBean and for schema initialization.
如果Quartz框架使用除了应用程序主数据源之外的其他数据源,需要声明一个数据源BEAN,并使用@QuartzDataSource注解标注它的@Bean方法,这样才可以确保SchedulerFactoryBean创建以及初始化时,都使用特定于QUARTZ制定的数据源。
@QuartzDataSource注解是SpringBoot提供的与QUARTZ集成时使用指定数据的方案

6. 如何禁止并发执行相同定义的JOBDETAIL?

(1). 场景描述

Quartz定时任务默认是并发执行的,不会等待上一次任务执行完毕再执行下一个任务,当到达间隔时间就会执行对应的任务,如果定时任执行太长,会长时间占用资源导致其它任务堵塞,同时会出现多个线程并存的情况。如何解决呢? 可以在JOB的实现类上添加@DisallowConcurrentExecution注解,禁止并发执行。

(2). @DisallowConcurrentExecution注解的含义是什么?

@DisallowConcurrentExecution注解,用于禁止并发执行多个相同定义的JobDetail。该注解添加在JOB的实现类上的,不是不能同时执行多个JOB,而是不能并发执行同一个Job Definition(由JobDetail定义),允许同时执行多个不同的JobDetail。如:
一个任务设定的时间间隔为3秒,但该任务的执行时间是5秒,当在JOB实现类上添加@DisallowConcurrentExecution注解以后,程序会等待上一个周期的任务执行完毕以后再去执行,如果不添加该注解,则会在3秒时再启用新的线程执行该任务。

源码、示例及DEMO

源码、示例及DEMO

  • 2
    点赞
  • 7
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
SpringBoot 中实现定时任务可以使用 Quartz 框架Quartz 是一个开源的作业调度框架,可以用来创建简单或复杂的作业调度程序。 下面是使用 Quartz 实现定时任务的步骤: 1. 添加 Quartz 的依赖 在 pom.xml 文件中添加以下依赖: ```xml <dependency> <groupId>org.quartz-scheduler</groupId> <artifactId>quartz</artifactId> <version>2.3.0</version> </dependency> ``` 2. 创建 Job 类 创建一个实现 org.quartz.Job 接口的 Job 类。Job 接口只有一个方法 execute(JobExecutionContext context),该方法会在作业执行时被调用。 ```java public class MyJob implements Job { @Override public void execute(JobExecutionContext context) throws JobExecutionException { // 作业要执行的任务 } } ``` 3. 创建 Trigger Trigger 是定义作业调度时间的组件。可以创建多个 Trigger 对象,每个对象对应一个时间表达式。 ```java Trigger trigger = TriggerBuilder.newTrigger() .withIdentity("triggerName", "groupName") .withSchedule(CronScheduleBuilder.cronSchedule("0/5 * * * * ?")) .build(); ``` 上面的代码创建了一个 Trigger 对象,它的名称是 triggerName,所属的组是 groupName。它使用了一个 Cron 表达式,表示每隔 5 秒执行一次作业。 4. 创建 Scheduler Scheduler 是 Quartz 的核心组件,用于管理和调度作业和触发器。创建一个 Scheduler 对象,然后将 Job 和 Trigger 注册到 Scheduler 中。 ```java SchedulerFactory schedulerFactory = new StdSchedulerFactory(); Scheduler scheduler = schedulerFactory.getScheduler(); JobDetail jobDetail = JobBuilder.newJob(MyJob.class) .withIdentity("jobName", "groupName") .build(); scheduler.scheduleJob(jobDetail, trigger); scheduler.start(); ``` 上面的代码创建了一个 Scheduler 对象,并使用 JobBuilder 和 TriggerBuilder 创建了一个 JobDetail 对象和一个 Trigger 对象。然后将 JobDetail 和 Trigger 注册到 Scheduler 中,并启动 Scheduler。 完整的代码示例: ```java public class MyJob implements Job { @Override public void execute(JobExecutionContext context) throws JobExecutionException { // 作业要执行的任务 } } public class QuartzConfig { @Bean public JobDetail jobDetail() { return JobBuilder.newJob(MyJob.class) .withIdentity("jobName", "groupName") .build(); } @Bean public Trigger trigger() { return TriggerBuilder.newTrigger() .withIdentity("triggerName", "groupName") .withSchedule(CronScheduleBuilder.cronSchedule("0/5 * * * * ?")) .build(); } @Bean public Scheduler scheduler() throws SchedulerException { SchedulerFactory schedulerFactory = new StdSchedulerFactory(); Scheduler scheduler = schedulerFactory.getScheduler(); scheduler.scheduleJob(jobDetail(), trigger()); scheduler.start(); return scheduler; } } ``` 在上面的代码中,我们使用了 Spring 的注解 @Bean 来注册 JobDetail、Trigger 和 Scheduler。注意要捕获 SchedulerException 异常。 这样就完成了通过集成 Quartz 来实现 SpringBoot 定时任务的步骤。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

cloneme01

谢谢您的支持与鼓励!

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值