zoukankan      html  css  js  c++  java
  • Quartz —— 任务调度框架

    一、Quartz

      Quartz  OpenSymphony 开源组织在任务调度领域的一个开源项目,完全基于 Java 实现。该项目于 2009 年被 Terracotta 收购,目前是 Terracotta 旗下的一个项目.

      主页 : http://www.quartz-scheduler.org/

      Github : https://github.com/quartz-scheduler/quartz

      1、特点

    • 强大的调度功能,例如支持丰富多样的调度方法,可以满足各种常规及特殊需求;
    • 灵活的应用方式,例如支持任务和调度的多种组合方式,支持调度数据的多种存储方式;
    • 分布式和集群能力

      2、专用词汇

    • scheduler : 是一个计划调度器容器,容器里面可以盛放众多的JobDetailtrigger,当容器启动后,里面的每个JobDetail都会根据trigger按部就班自动去执行。容器中有一个线程池,用来并行调度执行每个作业,这样可以提高容器效率。
    • trigger : 触发器,用于定义任务调度时间规则 ,即什么时候去调。
    • job : 任务,是一个可执行的工作,它本身可能是有状态的。
    • misfire : 错过的,指本来应该被执行但实际没有被执行的任务调度。
    • JobDetailTriggerscheduler容器上注册后,形成了装配好的作业(JobDetailTrigger所组成的一对儿),就可以伴随容器启动而调度执行了。

    二、入门案例

      本案例基于spring和quartz整合完成。

    1、导入依赖

     1 <dependencies>
     2         <!-- spring 框架 -->
     3         <!-- spring框架基础坐标 -->
     4         <dependency>
     5             <groupId>org.springframework</groupId>
     6             <artifactId>spring-context</artifactId>
     7             <version>4.2.4.RELEASE</version>
     8         </dependency>
     9         <!--扩展包 -->
    10         <dependency>
    11             <groupId>org.springframework</groupId>
    12             <artifactId>spring-context-support</artifactId>
    13             <version>4.2.4.RELEASE</version>
    14         </dependency>
    15         <!-- 事务包 -->
    16         <dependency>
    17             <groupId>org.springframework</groupId>
    18             <artifactId>spring-tx</artifactId>
    19             <version>4.2.4.RELEASE</version>
    20         </dependency>
    21 
    22         <!-- Quartz的坐标-->
    23         <dependency>
    24             <groupId>org.quartz-scheduler</groupId>
    25             <artifactId>quartz</artifactId>
    26             <version>2.2.3</version>
    27         </dependency>
    28         <dependency>
    29             <groupId>org.quartz-scheduler</groupId>
    30             <artifactId>quartz-jobs</artifactId>
    31             <version>2.2.3</version>
    32         </dependency>
    33 
    34         <dependency>
    35             <groupId>junit</groupId>
    36             <artifactId>junit</artifactId>
    37             <version>3.8.1</version>
    38             <scope>test</scope>
    39         </dependency>
    40     </dependencies>

    2、创建任务类

    1 public class MyJob {
    2     public void  run(){
    3         System.out.println("任务执行了---------"+
    4                 new SimpleDateFormat("yyyy-MM-dd HH:mm:ss").format(new Date()));
    5     }
    6 }

    3、在spring配置文件(applicationContext.xml)中配置相关内容

     1     <!-- 一.配置任务类-->
     2     <bean id="myJob" class="com.cenobitor.jobs.MyJob"/>
     3 
     4     <!-- 二.配置jobDetail-->
     5     <bean id="myJobDetail" class="org.springframework.scheduling.quartz.MethodInvokingJobDetailFactoryBean">
     6         <!-- 用的Job实例 -->
     7         <property name="targetObject">
     8             <ref bean="myJob"/>
     9         </property>
    10         <!--Job实例中的方法-->
    11         <property name="targetMethod">
    12             <value>run</value>
    13         </property>
    14     </bean>
    15     <!-- 五.配置触发器Trigger -->
    16     <bean name="myTrigger" class="org.springframework.scheduling.quartz.CronTriggerFactoryBean">
    17         <property name="jobDetail">
    18             <ref bean="myJobDetail"/>
    19         </property>
    20         <property name="cronExpression">
    21             <!-- 每5秒执行一次 -->
    22             <value>0/5 * * * * ?</value>
    23         </property>
    24     </bean>
    25 
    26     <!-- 六.配置scheduler工厂 -->
    27     <bean id="scheduler" class="org.springframework.scheduling.quartz.SchedulerFactoryBean">
    28         <property name="triggers">
    29             <ref bean="myTrigger"/>
    30         </property>
    31     </bean>

    4、加载spring配置文件,创建spring工厂

    1 public class App {
    2     public static void main( String[] args ) {
    3 
    4         ClassPathXmlApplicationContext classPathXmlApplicationContext =
    5                 new ClassPathXmlApplicationContext("applicationContext.xml");
    6     }
    7 }

    5、运行结果

    INFO: Starting Quartz Scheduler now
    任务执行了---------2018-03-29 17:27:05
    任务执行了---------2018-03-29 17:27:10
    任务执行了---------2018-03-29 17:27:15
    任务执行了---------2018-03-29 17:27:20

    三、cron 表达式的格式

      Quartz cron 表达式的格式十分类似于 UNIX cron 格式,但还是有少许明显的区别。区别之一就是 Quartz 的格式向下支持到秒级别的计划,而 UNIX cron 计划仅支持至分钟级。许多我们的触发计划要基于秒级递增的(例如,每45),因此这是一个非常好的差异。
      在 UNIX cron 里,要执行的作业(或者说命令)是存放在 cron 表达式中的,在第六个域位置上。Quartz  cron 表达式存放执行计划。引用了 cron 表达式的 CronTrigger 在计划的时间里会与 job 关联上。
      另一个与 UNIX cron 表达式的不同点是在表达式中支持域的数目。UNIX 给出五个域(分、时、日、月和周)Quartz 提供七个域。表 5.1 列出了 Quartz cron 表达式支持的七个域。

      Quartz Cron 表达式支持到七个域

    名称

    是否必须

    允许值

    特殊字符

    0-59

    , - * /

    0-59

    , - * /

    0-23

    , - * /

    1-31

    , - * ? / L W C

    1-12 或 JAN-DEC

    , - * /

    1-7 或 SUN-SAT

    , - * ? / L C #

    空 或 1970-2099

    , - * /

      月份和星期的名称是不区分大小写的。FRI  fri 是一样的。 
      域之间有空格分隔,这和 UNIX cron 一样。无可争辩的,我们能写的最简单的表达式看起来就是这个了: 

    * * * ? * * 

      这个表达会每秒钟(每分种的、每小时的、每天的)激发一个部署的 job 

    • ·理解特殊字符   

      同 UNIX cron 一样,Quartz cron 表达式支持用特殊字符来创建更为复杂的执行计划。然而,Quartz 在特殊字符的支持上比标准 UNIX cron 表达式更丰富了。 

    • * 星号 

      使用星号(*) 指示着你想在这个域上包含所有合法的值。例如,在月份域上使用星号意味着每个月都会触发这个 trigger

     
     表达式样例:     

    0 * 17 * * ? 

      意义:每天从下午5点到下午5:59中的每分钟激发一次 trigger。它停在下午 5:59 是因为值 17 在小时域上,在下午 6 点时,小时变为 18 了,也就不再理会这个 trigger,直到下一天的下午5点。 
      在你希望 trigger 在该域的所有有效值上被激发时使用 * 字符。 

    • ? 问号 
      ? 号只能用在日和周域上,但是不能在这两个域上同时使用。你可以认为 ? 字符是 "我并不关心在该域上是什么值。" 这不同于星号,星号是指示着该域上的每一个值。? 是说不为该域指定值。 
      不能同时这两个域上指定值的理由是难以解释甚至是难以理解的。基本上,假定同时指定值的话,意义就会变得含混不清了:考虑一下,如果一个表达式在日域上有值11,同时在周域上指定了 WED。那么是要 trigger 仅在每个月的11号,且正好又是星期三那天被激发?还是在每个星期三的11号被激发呢?要去除这种不明确性的办法就是不能同时在这两个域上指定值。 
      只要记住,假如你为这两域的其中一个指定了值,那就必须在另一个字值上放一个 ? 
      表达式样例: 
    0 10,44 14 ? 3 WEN

      意义:在三月中的每个星期三的下午 2:10:00  下午 2:44:00 被触发。 

    • , 逗号 
      逗号 (,) 是用来在给某个域上指定一个值列表的。例如,使用值 0,15,30,45 在秒域上意味着每15秒触发一个 trigger。 
      表达式样例: 
    0 0,15,30,45 * * * ? 

      意义:每刻钟触发一次 trigger 

    • /斜杠

      斜杠 (/)是用于时间表的递增的。我们刚刚用了逗号来表示每15分钟的递增,但是我们也能写成这样 0/15。

      表达式样例:

    0/15 0/30 * ** ?

      意义:在整点和半点时每15秒触发 trigger。

    • -中划线
      中划线 (-) 用于指定一个范围。例如,在小时域上的 3-8 意味着 "3,4,5,6,7  8 点。 域的值不允许回卷,所以像 50-10 这样的值是不允许的。 
      表达式样例: 
    0 45 3-8 ? * * 

      意义:在上午的3点至上午的8点的45分时触发 trigger 

    • L 字母Last

      L 明了某域上允许的最后一个值。它仅被日和周域支持。当用在日域上,表示的是在月域上指定的月份的最后一天。例如,当月域上指定了 JAN 时,在日域上的 L 会促使 trigger 131号被触发。假如月域上 SEP,那么 L 会预示着在930号触发。换句话说,就是不管指定了哪个月,都是在相应月份的时最后一天触发 trigger 
      表达式:

    0 0 8 L * ? 

      意义是在每个月最后一天的上午 8:00 触发 trigger在月域上 * 说明是 "每个月" 
      当 L 字母用于周域上,指示着周的最后一天,就是星期六 (或者数字7)。所以如果你需要在每个月的最后一个星期六下午的 11:59 触发 trigger,你可以用这样的表达式 :

    0 59 23 ? * L

      当使用于周域上,你可以用一个数字与 L 连起来表示月份的最后一个星期 X。例如,表达式:

    0 0 12 ? * 2L 

      说的是在每个月的最后一个星期一触发 trigger。  

    不要让范围和列表值与 L 连用

    虽然你能用星期数(1-7)与 L 连用,但是不允许你用一个范围值和列表值与 L 连用。这会产生不可预知的结果。

    •  W 字母 

         W 字符代表着工作日 (Mon-Fri),并且仅能用于日域中。它用来指定离指定日的最近的一个平日。大部分的商业处理都是基于工作周的,所以 W 字符可能是非常重要的。例如,日域中的15W 意味着 "离该月15号的最近一个平日。" 假如15号是星期六,那么 trigger 会在14(星期)触发,因为距15号最近的是星期一,这个例子中也会是17号(译者Unmi注:不会在17号触发的,如果是15W,可能会是在14(15号是星期六)或者15(15号是星期天)触发,也就是只能出现在邻近的一天,如果15号当天为平日直接就会当日执行)。W 只能用在指定的日域为单天,不能是范围或列表值。 

    • # 井号 
      # 字符仅能用于周域中。它用于指定月份中的第几周的哪一天。例如,如果你指定周域的值为6#3,它意思是某月的第三个周五 (6=星期五,#3意味着月份中的第三周)。另一个例子 2#1 意思是某月的第一个星期一 (2=星期一,#1意味着月份中的第一周)。注意,假如你指定 #5,然而月份中没有第 5 周,那么该月不会触发。

    示例:

    表达式意义:

    "0 0 12 * *?" 每天中午12点触发

    "0 15 10 ? **" 每天上午10:15触发

    "0 15 10 * *?" 每天上午10:15触发

    "0 15 10 * * ?*" 每天上午10:15触发

    "0 15 10 * * ?2005" 2005年的每天上午10:15触发

    "0 * 14 * *?" 在每天下午2点到下午2:59期间的每1分钟触发

    "0 0/5 14 * *?" 在每天下午2点到下午2:55期间的每5分钟触发

    "0 0/5 14,18 ** ?" 在每天下午2点到2:55期间和下午6点到6:55期间的每5分钟触发

    "0 0-5 14 * *?" 在每天下午2点到下午2:05期间的每1分钟触发

    "0 10,44 14 ? 3WED" 每年三月的星期三的下午2:102: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 ? * 6L2002-2005" 2002年至2005年的每月的最后一个星期五上午10:15触发

    "0 15 10 ? *6#3" 每月的第三个星期五上午10:15触发

  • 相关阅读:
    ABP框架系列之三十六:(MVC-Views-MVC视图)
    ABP框架系列之三十五:(MVC-Controllers-MVC控制器)
    ABP框架系列之三十四:(Multi-Tenancy-多租户)
    ABP框架系列之三十三:(Module-System-模块系统)
    2-kong的preserve_host和strip_uri解析
    2-Consul简介
    1-Consul系列文章
    1-Kong文章记录
    1-ES学些教程
    0-Jenkins系列学习文章
  • 原文地址:https://www.cnblogs.com/gdwkong/p/8670982.html
Copyright © 2011-2022 走看看