前言

  开心一刻

    晚上回家,爸妈正在吵架,见我回来就都不说话了,看见我妈坐在那里瞪着我爸,我就问老爸“你干什么了惹我妈生这么大气?”  我爸说“没有什么啊,倒是你,这么大了还没有媳妇,要是你有媳妇给我们生一个孙子玩,我们致于吵架吗?”我一听就感觉要坏,老爸你这是来了一招调虎离山啊,实力坑儿子啊,果然我妈改瞪我了,然后完全不理我爸,直接指着我开骂了……

  路漫漫其修远兮,吾将上下而求索!

  github:https://github.com/youzhibing

  码云(gitee):https://gitee.com/youzhibing

java定时任务调度的实现方式 

  Timer

    这个相信大家都有用过,我也用过,但用的不多;

    特点是:简单易用,但由于所有任务都是由同一个线程来调度,因此所有任务都是串行执行的,同一时间只能有一个任务在执行,前一个任务的延迟或异常都将会影响到之后的任务;能实现简单的定时任务,稍微复杂点(或要求高一些)的定时任务却不好实现。
  ScheduledExecutor

    这个我相信大家也都用过,而且用的比Timer多;正是鉴于Timer的缺陷,Java 5推出了基于线程池设计的ScheduledExecutor;

    特点:每一个被调度的任务都会由线程池中一个线程去执行,因此任务是并发执行的,相互之间不会受到干扰。需要注意的是,只有当任务的执行时间到来时,ScheduedExecutor 才会真正启动一个线程,其余时间 ScheduledExecutor 都是在轮询任务的状态。

    虽然用ScheduledExecutor和Calendar能够实现复杂任务调度,但实现起来还是比较麻烦,对开发还是不够友善。

  Spring Scheduler

    spring对任务调度的实现支持,可以指定任务的执行时间,但对任务队列和线程池的管控较弱;一般集成于项目中,小任务很方便。
  JCronTab

    JCronTab则是一款完全按照crontab语法编写的java任务调度工具。

    特点:

      可指定任务的执行时间;

      提供完全按照Unix的UNIX-POSIX crontab的格式来规定时间;

      支持多种任务调度的持久化方法,包括普通文件、数据库以及 XML 文件进行持久化;

      JCronTab内置了发邮件功能,可以将任务执行结果方便地发送给需要被通知的人;

      设计和部署是高性能并可扩展。

  Quartz

    本文主角,请往下看
  当然还有XXL-JOB、Elastic-Job、Saturn等等

quartz相关概念

  Scheduler:调度器,进行任务调度;quartz的大脑
  Job:业务job,亦可称业务组件;定时任务的具体执行业务需要实现此接口,调度器会调用此接口的execute方法完成我们的定时业务
  JobDetail:用来定义业务Job的实例,我们可以称之为quartz job,很多时候我们谈到的job指的是JobDetail
  Trigger:触发器,用来定义一个指定的Job何时被执行
  JobBuilder:Job构建器,用来定义或创建JobDetail的实例;JobDetail限定了只能是Job的实例
  TriggerBuilder:触发器构建器,用来定义或创建触发器的实例

  具体为什么要分这么细,大家可以去查阅下相关资料,你会发现很多东西

工程实现

  pom.xml

  1. <?xml version="1.0" encoding="UTF-8"?>
  2. <project xmlns="http://maven.apache.org/POM/4.0.0"
  3. xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
  4. xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
  5. <modelVersion>4.0.0</modelVersion>
  6.  
  7. <groupId>com.lee</groupId>
  8. <artifactId>spring-boot-quartz</artifactId>
  9. <version>1.0-SNAPSHOT</version>
  10.  
  11. <properties>
  12. <java.version>1.8</java.version>
  13. <maven.compiler.source>1.8</maven.compiler.source>
  14. <maven.compiler.target>1.8</maven.compiler.target>
  15. <druid.version>1.1.10</druid.version>
  16. <pagehelper.version>1.2.5</pagehelper.version>
  17. <druid.version>1.1.10</druid.version>
  18. </properties>
  19.  
  20. <parent>
  21. <groupId>org.springframework.boot</groupId>
  22. <artifactId>spring-boot-starter-parent</artifactId>
  23. <version>2.0.3.RELEASE</version>
  24. </parent>
  25.  
  26. <dependencies>
  27. <dependency>
  28. <groupId>org.springframework.boot</groupId>
  29. <artifactId>spring-boot-starter-web</artifactId>
  30. </dependency>
  31. <dependency>
  32. <groupId>org.springframework.boot</groupId>
  33. <artifactId>spring-boot-starter-thymeleaf</artifactId>
  34. </dependency>
  35.  
  36. <dependency>
  37. <groupId>org.springframework.boot</groupId>
  38. <artifactId>spring-boot-starter-quartz</artifactId>
  39. </dependency>
  40.  
  41. <dependency>
  42. <groupId>com.alibaba</groupId>
  43. <artifactId>druid-spring-boot-starter</artifactId>
  44. <version>${druid.version}</version>
  45. </dependency>
  46. <dependency>
  47. <groupId>mysql</groupId>
  48. <artifactId>mysql-connector-java</artifactId>
  49. </dependency>
  50. <dependency>
  51. <groupId>com.github.pagehelper</groupId>
  52. <artifactId>pagehelper-spring-boot-starter</artifactId>
  53. <version>${pagehelper.version}</version>
  54. </dependency>
  55.  
  56. <!-- 日志 -->
  57. <dependency>
  58. <groupId>org.springframework.boot</groupId>
  59. <artifactId>spring-boot-starter-logging</artifactId>
  60. <exclusions> <!-- 排除spring-boot-starter-logging中的全部依赖 -->
  61. <exclusion>
  62. <groupId>*</groupId>
  63. <artifactId>*</artifactId>
  64. </exclusion>
  65. </exclusions>
  66. <scope>test</scope> <!-- 打包的时候不打spring-boot-starter-logging.jar -->
  67. </dependency>
  68. <dependency>
  69. <groupId>ch.qos.logback</groupId>
  70. <artifactId>logback-classic</artifactId>
  71. </dependency>
  72.  
  73. <dependency>
  74. <groupId>org.projectlombok</groupId>
  75. <artifactId>lombok</artifactId>
  76. <optional>true</optional>
  77. </dependency>
  78. </dependencies>
  79.  
  80. <build>
  81. <finalName>spring-boot-quartz</finalName>
  82. <plugins>
  83. <!-- 打包项目 mvn clean package -->
  84. <plugin>
  85. <groupId>org.springframework.boot</groupId>
  86. <artifactId>spring-boot-maven-plugin</artifactId>
  87. </plugin>
  88. </plugins>
  89. </build>
  90. </project>

  application.xml

  1. server:
  2. port: 9001
  3. servlet:
  4. context-path: /quartz
  5. spring:
  6. thymeleaf:
  7. mode: HTML
  8. cache: false
  9. #连接池配置
  10. datasource:
  11. type: com.alibaba.druid.pool.DruidDataSource
  12. druid:
  13. driver-class-name: com.mysql.jdbc.Driver
  14. url: jdbc:mysql://localhost:3306/spring-boot-quartz?useSSL=false&useUnicode=true
  15. username: root
  16. password: 123456
  17. initial-size: 1 #连接池初始大小
  18. max-active: 20 #连接池中最大的活跃连接数
  19. min-idle: 1 #连接池中最小的活跃连接数
  20. max-wait: 60000 #配置获取连接等待超时的时间
  21. pool-prepared-statements: true #打开PSCache,并且指定每个连接上PSCache的大小
  22. max-pool-prepared-statement-per-connection-size: 20
  23. validation-query: SELECT 1 FROM DUAL
  24. validation-query-timeout: 30000
  25. test-on-borrow: false #是否在获得连接后检测其可用性
  26. test-on-return: false #是否在连接放回连接池后检测其可用性
  27. test-while-idle: true #是否在连接空闲一段时间后检测其可用性
  28. quartz:
  29. #相关属性配置
  30. properties:
  31. org:
  32. quartz:
  33. scheduler:
  34. instanceName: quartzScheduler
  35. instanceId: AUTO
  36. jobStore:
  37. class: org.quartz.impl.jdbcjobstore.JobStoreTX
  38. driverDelegateClass: org.quartz.impl.jdbcjobstore.StdJDBCDelegate
  39. tablePrefix: QRTZ_
  40. isClustered: false
  41. clusterCheckinInterval: 10000
  42. useProperties: false
  43. threadPool:
  44. class: org.quartz.simpl.SimpleThreadPool
  45. threadCount: 10
  46. threadPriority: 5
  47. threadsInheritContextClassLoaderOfInitializingThread: true
  48. #数据库方式
  49. job-store-type: JDBC
  50. #初始化表结构
  51. jdbc:
  52. initialize-schema: NEVER
  53. #mybatis配置
  54. mybatis:
  55. type-aliases-package: com.lee.quartz.entity
  56. mapper-locations: classpath:mybatis/mapper/*.xml
  57. #分页配置, pageHelper是物理分页插件
  58. pagehelper:
  59. #4.0.0以后版本可以不设置该参数,该示例中是5.1.4
  60. helper-dialect: mysql
  61. #启用合理化,如果pageNum<1会查询第一页,如果pageNum>pages会查询最后一页
  62. reasonable: true
  63. logging:
  64. level:
  65. com.lee.quartz.mapper: debug

  这样,quartz就配置好了,应用里面直接用即可

  JobController.java

  1. package com.lee.quartz.web;
  2.  
  3. import com.github.pagehelper.PageInfo;
  4. import com.lee.quartz.common.Result;
  5. import com.lee.quartz.entity.QuartzJob;
  6. import com.lee.quartz.service.IJobService;
  7. import org.slf4j.Logger;
  8. import org.slf4j.LoggerFactory;
  9. import org.springframework.beans.factory.annotation.Autowired;
  10. import org.springframework.web.bind.annotation.PostMapping;
  11. import org.springframework.web.bind.annotation.RequestMapping;
  12. import org.springframework.web.bind.annotation.RestController;
  13.  
  14. @RestController
  15. @RequestMapping("/job")
  16. public class JobController {
  17. private final static Logger LOGGER = LoggerFactory.getLogger(JobController.class);
  18.  
  19. @Autowired
  20. private IJobService jobService;
  21.  
  22. @SuppressWarnings({ "unchecked", "rawtypes" })
  23. @PostMapping("/add")
  24. public Result save(QuartzJob quartz){
  25. LOGGER.info("新增任务");
  26. Result result = jobService.saveJob(quartz);
  27. return result;
  28. }
  29. @PostMapping("/list")
  30. public PageInfo list(String jobName,Integer pageNo,Integer pageSize){
  31. LOGGER.info("任务列表");
  32. PageInfo pageInfo = jobService.listQuartzJob(jobName, pageNo, pageSize);
  33. return pageInfo;
  34. }
  35.  
  36. @PostMapping("/trigger")
  37. public Result trigger(String jobName, String jobGroup) {
  38. LOGGER.info("触发任务");
  39. Result result = jobService.triggerJob(jobName, jobGroup);
  40. return result;
  41. }
  42.  
  43. @PostMapping("/pause")
  44. public Result pause(String jobName, String jobGroup) {
  45. LOGGER.info("停止任务");
  46. Result result = jobService.pauseJob(jobName, jobGroup);
  47. return result;
  48. }
  49.  
  50. @PostMapping("/resume")
  51. public Result resume(String jobName, String jobGroup) {
  52. LOGGER.info("恢复任务");
  53. Result result = jobService.resumeJob(jobName, jobGroup);
  54. return result;
  55. }
  56.  
  57. @PostMapping("/remove")
  58. public Result remove(String jobName, String jobGroup) {
  59. LOGGER.info("移除任务");
  60. Result result = jobService.removeJob(jobName, jobGroup);
  61. return result;
  62. }
  63. }

  JobServiceImpl.java

  1. package com.lee.quartz.service.impl;
  2.  
  3. import com.github.pagehelper.PageHelper;
  4. import com.github.pagehelper.PageInfo;
  5. import com.lee.quartz.common.Result;
  6. import com.lee.quartz.entity.QuartzJob;
  7. import com.lee.quartz.mapper.JobMapper;
  8. import com.lee.quartz.service.IJobService;
  9. import org.quartz.*;
  10. import org.springframework.beans.factory.annotation.Autowired;
  11. import org.springframework.stereotype.Service;
  12.  
  13. import java.util.List;
  14.  
  15. @Service
  16. public class JobServiceImpl implements IJobService {
  17.  
  18. @Autowired
  19. private Scheduler scheduler;
  20.  
  21. @Autowired
  22. private JobMapper jobMapper;
  23.  
  24. @Override
  25. public PageInfo listQuartzJob(String jobName, Integer pageNum, Integer pageSize) {
  26. PageHelper.startPage(pageNum, pageSize);
  27. List<QuartzJob> jobList = jobMapper.listJob(jobName);
  28. PageInfo pageInfo = new PageInfo(jobList);
  29. return pageInfo;
  30. }
  31.  
  32. @Override
  33. public Result saveJob(QuartzJob quartz){
  34. try {
  35. //如果是修改 展示旧的 任务
  36. if(quartz.getOldJobGroup() != null && !"".equals(quartz.getOldJobGroup())){
  37. JobKey key = new JobKey(quartz.getOldJobName(),quartz.getOldJobGroup());
  38. scheduler.deleteJob(key);
  39. }
  40.  
  41. //构建job信息
  42. Class cls = Class.forName(quartz.getJobClassName()) ;
  43. cls.newInstance();
  44. JobDetail job = JobBuilder.newJob(cls).withIdentity(quartz.getJobName(),
  45. quartz.getJobGroup())
  46. .withDescription(quartz.getDescription()).build();
  47. // 触发时间点
  48. CronScheduleBuilder cronScheduleBuilder = CronScheduleBuilder.cronSchedule(quartz.getCronExpression().trim());
  49. Trigger trigger = TriggerBuilder.newTrigger().withIdentity("trigger"+quartz.getJobName(), quartz.getJobGroup())
  50. .startNow().withSchedule(cronScheduleBuilder).build();
  51. //交由Scheduler安排触发
  52. scheduler.scheduleJob(job, trigger);
  53. } catch (Exception e) {
  54. e.printStackTrace();
  55. return Result.error();
  56. }
  57. return Result.ok();
  58. }
  59.  
  60. @Override
  61. public Result triggerJob(String jobName, String jobGroup) {
  62. JobKey key = new JobKey(jobName,jobGroup);
  63. try {
  64. scheduler.triggerJob(key);
  65. } catch (SchedulerException e) {
  66. e.printStackTrace();
  67. return Result.error();
  68. }
  69. return Result.ok();
  70. }
  71.  
  72. @Override
  73. public Result pauseJob(String jobName, String jobGroup) {
  74. JobKey key = new JobKey(jobName,jobGroup);
  75. try {
  76. scheduler.pauseJob(key);
  77. } catch (SchedulerException e) {
  78. e.printStackTrace();
  79. return Result.error();
  80. }
  81. return Result.ok();
  82. }
  83.  
  84. @Override
  85. public Result resumeJob(String jobName, String jobGroup) {
  86. JobKey key = new JobKey(jobName,jobGroup);
  87. try {
  88. scheduler.resumeJob(key);
  89. } catch (SchedulerException e) {
  90. e.printStackTrace();
  91. return Result.error();
  92. }
  93. return Result.ok();
  94. }
  95.  
  96. @Override
  97. public Result removeJob(String jobName, String jobGroup) {
  98. try {
  99. TriggerKey triggerKey = TriggerKey.triggerKey(jobName, jobGroup);
  100. // 停止触发器
  101. scheduler.pauseTrigger(triggerKey);
  102. // 移除触发器
  103. scheduler.unscheduleJob(triggerKey);
  104. // 删除任务
  105. scheduler.deleteJob(JobKey.jobKey(jobName, jobGroup));
  106. System.out.println("removeJob:"+JobKey.jobKey(jobName));
  107. } catch (Exception e) {
  108. e.printStackTrace();
  109. return Result.error();
  110. }
  111. return Result.ok();
  112. }
  113. }

  主要就是以上文件,详情请查看spring-boot-quartz

  工程里面数据源用的druid,springboot默认也会将该数据源应用到quartz,如果想给quartz单独配置数据源,可配合@QuartzDataSource来实现(更多quarz数据源问题,请查看spring-boot-2.0.3之quartz集成,数据源问题,源码探究

  最终效果如下

trigger状态

  org.quartz.impl.jdbcjobstore.Constants中存放了一些列的常量,源代码如下

  1. /*
  2. * All content copyright Terracotta, Inc., unless otherwise indicated. All rights reserved.
  3. *
  4. * Licensed under the Apache License, Version 2.0 (the "License"); you may not
  5. * use this file except in compliance with the License. You may obtain a copy
  6. * of the License at
  7. *
  8. * http://www.apache.org/licenses/LICENSE-2.0
  9. *
  10. * Unless required by applicable law or agreed to in writing, software
  11. * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
  12. * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
  13. * License for the specific language governing permissions and limitations
  14. * under the License.
  15. *
  16. */
  17.  
  18. package org.quartz.impl.jdbcjobstore;
  19.  
  20. /**
  21. * <p>
  22. * This interface can be implemented by any <code>{@link
  23. * org.quartz.impl.jdbcjobstore.DriverDelegate}</code>
  24. * class that needs to use the constants contained herein.
  25. * </p>
  26. *
  27. * @author <a href="mailto:jeff@binaryfeed.org">Jeffrey Wescott</a>
  28. * @author James House
  29. */
  30. public interface Constants {
  31.  
  32. /*
  33. * ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
  34. *
  35. * Constants.
  36. *
  37. * ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
  38. */
  39.  
  40. // Table names
  41. String TABLE_JOB_DETAILS = "JOB_DETAILS";
  42.  
  43. String TABLE_TRIGGERS = "TRIGGERS";
  44.  
  45. String TABLE_SIMPLE_TRIGGERS = "SIMPLE_TRIGGERS";
  46.  
  47. String TABLE_CRON_TRIGGERS = "CRON_TRIGGERS";
  48.  
  49. String TABLE_BLOB_TRIGGERS = "BLOB_TRIGGERS";
  50.  
  51. String TABLE_FIRED_TRIGGERS = "FIRED_TRIGGERS";
  52.  
  53. String TABLE_CALENDARS = "CALENDARS";
  54.  
  55. String TABLE_PAUSED_TRIGGERS = "PAUSED_TRIGGER_GRPS";
  56.  
  57. String TABLE_LOCKS = "LOCKS";
  58.  
  59. String TABLE_SCHEDULER_STATE = "SCHEDULER_STATE";
  60.  
  61. // TABLE_JOB_DETAILS columns names
  62.  
  63. String COL_SCHEDULER_NAME = "SCHED_NAME";
  64.  
  65. String COL_JOB_NAME = "JOB_NAME";
  66.  
  67. String COL_JOB_GROUP = "JOB_GROUP";
  68.  
  69. String COL_IS_DURABLE = "IS_DURABLE";
  70.  
  71. String COL_IS_VOLATILE = "IS_VOLATILE";
  72.  
  73. String COL_IS_NONCONCURRENT = "IS_NONCONCURRENT";
  74.  
  75. String COL_IS_UPDATE_DATA = "IS_UPDATE_DATA";
  76.  
  77. String COL_REQUESTS_RECOVERY = "REQUESTS_RECOVERY";
  78.  
  79. String COL_JOB_DATAMAP = "JOB_DATA";
  80.  
  81. String COL_JOB_CLASS = "JOB_CLASS_NAME";
  82.  
  83. String COL_DESCRIPTION = "DESCRIPTION";
  84.  
  85. // TABLE_TRIGGERS columns names
  86. String COL_TRIGGER_NAME = "TRIGGER_NAME";
  87.  
  88. String COL_TRIGGER_GROUP = "TRIGGER_GROUP";
  89.  
  90. String COL_NEXT_FIRE_TIME = "NEXT_FIRE_TIME";
  91.  
  92. String COL_PREV_FIRE_TIME = "PREV_FIRE_TIME";
  93.  
  94. String COL_TRIGGER_STATE = "TRIGGER_STATE";
  95.  
  96. String COL_TRIGGER_TYPE = "TRIGGER_TYPE";
  97.  
  98. String COL_START_TIME = "START_TIME";
  99.  
  100. String COL_END_TIME = "END_TIME";
  101.  
  102. String COL_PRIORITY = "PRIORITY";
  103.  
  104. String COL_MISFIRE_INSTRUCTION = "MISFIRE_INSTR";
  105.  
  106. String ALIAS_COL_NEXT_FIRE_TIME = "ALIAS_NXT_FR_TM";
  107.  
  108. // TABLE_SIMPLE_TRIGGERS columns names
  109. String COL_REPEAT_COUNT = "REPEAT_COUNT";
  110.  
  111. String COL_REPEAT_INTERVAL = "REPEAT_INTERVAL";
  112.  
  113. String COL_TIMES_TRIGGERED = "TIMES_TRIGGERED";
  114.  
  115. // TABLE_CRON_TRIGGERS columns names
  116. String COL_CRON_EXPRESSION = "CRON_EXPRESSION";
  117.  
  118. // TABLE_BLOB_TRIGGERS columns names
  119. String COL_BLOB = "BLOB_DATA";
  120.  
  121. String COL_TIME_ZONE_ID = "TIME_ZONE_ID";
  122.  
  123. // TABLE_FIRED_TRIGGERS columns names
  124. String COL_INSTANCE_NAME = "INSTANCE_NAME";
  125.  
  126. String COL_FIRED_TIME = "FIRED_TIME";
  127.  
  128. String COL_SCHED_TIME = "SCHED_TIME";
  129.  
  130. String COL_ENTRY_ID = "ENTRY_ID";
  131.  
  132. String COL_ENTRY_STATE = "STATE";
  133.  
  134. // TABLE_CALENDARS columns names
  135. String COL_CALENDAR_NAME = "CALENDAR_NAME";
  136.  
  137. String COL_CALENDAR = "CALENDAR";
  138.  
  139. // TABLE_LOCKS columns names
  140. String COL_LOCK_NAME = "LOCK_NAME";
  141.  
  142. // TABLE_LOCKS columns names
  143. String COL_LAST_CHECKIN_TIME = "LAST_CHECKIN_TIME";
  144.  
  145. String COL_CHECKIN_INTERVAL = "CHECKIN_INTERVAL";
  146.  
  147. // MISC CONSTANTS
  148. String DEFAULT_TABLE_PREFIX = "QRTZ_";
  149.  
  150. // STATES
  151. String STATE_WAITING = "WAITING";
  152.  
  153. String STATE_ACQUIRED = "ACQUIRED";
  154.  
  155. String STATE_EXECUTING = "EXECUTING";
  156.  
  157. String STATE_COMPLETE = "COMPLETE";
  158.  
  159. String STATE_BLOCKED = "BLOCKED";
  160.  
  161. String STATE_ERROR = "ERROR";
  162.  
  163. String STATE_PAUSED = "PAUSED";
  164.  
  165. String STATE_PAUSED_BLOCKED = "PAUSED_BLOCKED";
  166.  
  167. String STATE_DELETED = "DELETED";
  168.  
  169. /**
  170. * @deprecated Whether a trigger has misfired is no longer a state, but
  171. * rather now identified dynamically by whether the trigger's next fire
  172. * time is more than the misfire threshold time in the past.
  173. */
  174. String STATE_MISFIRED = "MISFIRED";
  175.  
  176. String ALL_GROUPS_PAUSED = "_$_ALL_GROUPS_PAUSED_$_";
  177.  
  178. // TRIGGER TYPES
  179. /** Simple Trigger type. */
  180. String TTYPE_SIMPLE = "SIMPLE";
  181.  
  182. /** Cron Trigger type. */
  183. String TTYPE_CRON = "CRON";
  184.  
  185. /** Calendar Interval Trigger type. */
  186. String TTYPE_CAL_INT = "CAL_INT";
  187.  
  188. /** Daily Time Interval Trigger type. */
  189. String TTYPE_DAILY_TIME_INT = "DAILY_I";
  190.  
  191. /** A general blob Trigger type. */
  192. String TTYPE_BLOB = "BLOB";
  193. }
  194.  
  195. // EOF

  里面有quartz的表名、各个表包含的列名、trigger状态、trigger类型等内容

  状态包括

    WAITING:等待中
    ACQUIRED:将触发,此时还未到trigger真正的触发时刻
    EXECUTING:触发,亦可理解成执行中,trigger真正的触发时刻
    COMPLETE:完成,不再触发
    BLOCKED:受阻,不允许并发执行job时会出现(@DisallowConcurrentExecution)
    ERROR:出错
    PAUSED:暂停中
    PAUSED_BLOCKED:暂停受阻,不允许并发执行job时会出现(@DisallowConcurrentExecution)
    DELETED:已删除
    MISFIRED:触发失败,已弃用,有另外的替代方式

  状态变化流程图如下所示

  trigger的初始状态是WAITING,处于WAITING状态的trigger等待被触发。调度线程会不停地扫triggers表,根据NEXT_FIRE_TIME提前拉取即将触发的trigger,如果这个trigger被该调度线程拉取到,它的状态就会变为ACQUIRED。因为是提前拉取trigger,并未到达trigger真正的触发时刻,所以调度线程会等到真正触发的时刻,再将trigger状态由ACQUIRED改为EXECUTING。如果这个trigger不再执行,就将状态改为COMPLETE,否则为WAITING,开始新的周期。如果这个周期中的任何环节抛出异常,trigger的状态会变成ERROR。如果手动暂停这个trigger,状态会变成PAUSED。

总结

  Quartz作为一个开源的作业调度框架,提供了巨大的灵活性而不牺牲简单性。我们能够用它来为执行一个作业而创建简单的或复杂的调度。它有很多特征,如:数据库、集群、插件、JavaMail支持,EJB作业预构建,支持cron-like表达式等等;

  springboot集成quartz非常简单,最简单的情况下只需要引入依赖我们就可以享受quartz提供的功能,springboot默认会帮我们配置好quartz;当然我们也可以自定义配置来实现quartz的定制;

参考

  几种任务调度的Java实现方法与比较

  小柒2012 / spring-boot-quartz

  boot-features-quartz

  作业调度系统—Quartz

  记一次Quartz重复调度(任务重复执行)的问题排查

  Quartz FAQ

spring-boot-2.0.3之quartz集成,不是你想的那样哦!的更多相关文章

  1. Spring Boot 入门(九):集成Quartz定时任务

    本片文章续<Spring Boot 入门(八):集成RabbitMQ消息队列>,关于Quartz定时任务请参考<Quartz的基本使用之入门(2.3.0版本)> spring ...

  2. Spring Boot 2.0(八):Spring Boot 集成 Memcached

    Memcached 介绍 Memcached 是一个高性能的分布式内存对象缓存系统,用于动态Web应用以减轻数据库负载.它通过在内存中缓存数据和对象来减少读取数据库的次数,从而提高动态.数据库驱动网站 ...

  3. springboot2.0(一):【重磅】Spring Boot 2.0权威发布

    就在昨天Spring Boot2.0.0.RELEASE正式发布,今天早上在发布Spring Boot2.0的时候还出现一个小插曲,将Spring Boot2.0同步到Maven仓库的时候出现了错误, ...

  4. 业余草分享 Spring Boot 2.0 正式发布的新特性

    就在昨天Spring Boot2.0.0.RELEASE正式发布,今天早上在发布Spring Boot2.0的时候还出现一个小插曲,将Spring Boot2.0同步到Maven仓库的时候出现了错误, ...

  5. 【重磅】Spring Boot 2.0权威发布

    新版本特性 新版本值得关注的亮点有哪些: 基于 Java 8,支持 Java 9 也就是说Spring Boot2.0的最低版本要求为JDK8,据了解国内大部分的互联网公司系统都还跑在JDK1.6/7 ...

  6. spring-boot-2.0.3之quartz集成,最佳实践

    前言 开心一刻 快过年了,大街上,爷爷在给孙子示范摔炮怎么放,嘴里还不停念叨:要像这样,用劲甩才能响.示范了一个,两个,三个... 孙子终于忍不住了,抱着爷爷的腿哭起来:爷呀,你给我剩个吧! 新的一年 ...

  7. (转)Spring Boot 2(一):【重磅】Spring Boot 2.0权威发布

    http://www.ityouknow.com/springboot/2018/03/01/spring-boot-2.0.html 就在今天Spring Boot2.0.0.RELEASE正式发布 ...

  8. 重磅:Spring Boot 2.0 正式发布!

    Spring Boot 2.0 正式发布! 2018/03/01最新消息,传得沸沸扬扬的Spring Boot 2.0 正式发布了. 小编去看了下Spring Boot的官网,正式版本已经释放出来了! ...

  9. Spring Boot 2(一):Spring Boot 2.0新特性

    Spring Boot 2(一):Spring Boot 2.0新特性 Spring Boot依赖于Spring,而Spring Cloud又依赖于Spring Boot,因此Spring Boot2 ...

随机推荐

  1. Educational Codeforces Round 58 (Rated for Div. 2) F dp + 优化(新坑) + 离线处理

    https://codeforces.com/contest/1101/problem/F 题意 有n个城市,m辆卡车,每辆卡车有起点\(s_i\),终点\(f_i\),每公里油耗\(c_i\),可加 ...

  2. freeType移植总结①——使用keil编译freeType2库

    在各个技术博客搜索相关资料后,终于将freeType的源码用keil工程编译通过,这里记录一下步骤和遇到的问题. 因为网上的资料都是旧版本freeType的工程,这里博主使用的是freeType2.9 ...

  3. python 0228

    01 cpu 内存 硬盘 操作系统 CPU:中央处理器,相当于人大脑. 飞机 内存:临时存储数据. 8g,16g, 高铁 1,成本高. 2,断电即消失. 硬盘:长期存储大量的数据. 1T 512G等等 ...

  4. java中接口和继承的区别

    实际概念区别:区别1:不同的修饰符修饰(interface),(extends)区别2:在面向对象编程中可以有多继承!但是只支持接口的多继承,不支持'继承'的多继承哦而继承在java中具有单根性,子类 ...

  5. azure cosmos db (mongo DB)

    使用.net mongo的操作类操作azure(微软云)cosmosdb时,发现在做delete的操作的时候可以传一个文档对象,但是最后这个文档会解析成具体的sql语句,而当文档特别大时这样就出先了转 ...

  6. robotframework 远程连接数据库问题

    今天在使用RF远程连接数据库时出现报错的问题,因为我发现只安装databaselibrary跟PyMsql的话,连接本地的数据库是OK的,但是如果我们的测试机性能有限,那么既要编写代码.运行测试.还有 ...

  7. POJ - 3278 Catch That Cow bfs 线性

    #include<stdio.h> #include<string.h> #include<algorithm> #include<queue> usi ...

  8. git忽视修改的文件

    对于tracked文件来说,使用.gitignore是不行的.但是git提供了新的方法. 使用命令 git update-index --assume-unchanged <files>, ...

  9. HARD FAULT

    程序陷在while(1)里面 解决办法 定点到发生死循环的位置 打开stack windows逐层查找发生死循环之前运行过的函数 导致原因 1 内存溢出或者访问越界,通常为数组或结构体访问越界.这个需 ...

  10. Python之路【第五篇】函数

    4.1 函数的定义 函数是指将一组语句的集合通过一个名字(函数名)封装起来,要想执行这个函数,只需调用其函数名即可 4.2 函数的创建 函数名的命名规则: 1.函数名必须以下划线或字母开头,可以包含任 ...