Quartz Spring与Spring Task总结

    Spring Quartz 作了一个封装,同时, Spring 自己也提供了一个任务定时器 (spring-task) ,现把它总结一下。
     对于 Quartz ,我们使用的时候主要是注重两个方面,一个是定时任务的业务,另一个就是 Cron 表达式。定时任务跟具体的业务相关,这无需多说,这里只说明表达式含义及其写法。
     Cron 表达式包括下面 7 个字段并区别顺序 0-59 0-59 小时 0-23 月内日期 1-31 1-12 或者 JAN-DEC 周内日期 1-7 或者 SUN-SAT ( 可选字段 ) 留空或者 1970-2099 并且通过特殊字符表示特殊意义 具体为下
     斜线 (/) 字符表示增量值 例如 在秒字段中 "5/15" 代表从第 5 秒开始 15 秒一次
     问号 (?) 字符和字母 L 字符只有在月内日期和周内日期字段中可用 问号表示这个字段不包含具体值 所以 如果指定月内日期 可以在周内日期字段中插入 "?" 表示周内日期值无关紧要 这里有个很蛋疼的设定,无关 Quartz ,而是 Spring 集成 Quartz 后,它自己加的一个约束,那就是:日期 (1-31) 和星期 (SUN-SAT) 两者,必须有一个是问号 (?) ,系统在启动的时候, Spring 会检查表达式,如果不符合它的规则,就会抛异常。所以在使用的时候这个地方一定要注意,而这个在 Linux 上执行 Cron 是没有这个限制的。
     字母 L 字符是 last 的缩写 放在月内日期字段中 表示安排在当月最后一天执行 在周内日期字段中 如果 "L" 单独存在 , 就等于 "7" 否则代表当月内周内日期的最后一个实例 所以 "0L" 表示安排在当月的最后一个星期日执行
     字母 (W) 字符把执行安排在最靠近指定值的工作日 "1W" 放在月内日期字段中 表示把执行安排在当月的第一个工作日内
     井号 (#) 字符为给定月份指定具体的工作日实例 "MON#2" 放在周内日期字段中 表示把任务安排在当月的第二个星期一
     星号 (*) 字符是通配字符 , 表示该字段可以接受任何可能的值 表达式例子。
     例子:
     "00 08 * * ?" 每天上午 8 点触发
     "015 10 ? * *" 每天上午 10:15 触发
     "015 10 * * ?" 每天上午 10:15 触发
     "015 10 * * ? *" 每天上午 10:15 触发
     "0 15 10 * * ? 2005" 2005 年的每天上午 10:15 触发
     "0* 14 * * ?" 在每天下午 2 点到下午 2:59 期间的每 1 分钟触发
     "00/5 14 * * ?" 在每天下午 2 点到下午 2:55 期间的每 5 分钟触发
     "00/5 14,18 * * ?" 在每天下午 2 点到 2:55 期间和下午 6 点到 6:55 期间的每 5 分钟触发
     "00-5 14 * * ?" 在每天下午 2 点到下午 2:05 期间的每 1 分钟触发
     "010,44 14 ? 3 WED" 每年三月的星期三的下午 2:10 2:44 触发
     "015 10 ? * MON-FRI" 周一至周五的上午 10:15 触发
     "015 10 15 * ?" 每月 15 日上午 10:15 触发
     "015 10 L * ?" 每月最后一日的上午 10:15 触发
     "015 10 ? * 6L" 每月的最后一个星期五上午 10:15 触发
     "015 10 ? * 6L 2009-2019" 2009 年至 2019 年的每月的最后一个星期五上午 10:15 触发
     "015 10 ? * 6#3" 每月的第三个星期五上午 10:15 触发

     使用 Spring Quartz 实现 Job 任务有两种方式,一种是继承 org.springframework.scheduling.quartz.QuartzJobBean ,这个不推荐。 另一种不需要继承,只需要在配置文件中定义 org.springframework.scheduling.quartz.MethodInvokingJobDetailFactoryBean ,并指定它的 targetObject 属性为 Job 任务类, targetMethod 属性为任务方法就可以了。
    < bean id= "job" class= " xx.xx.xx.Job" />
    < bean id= "cronTask" class= "org.springframework.scheduling.quartz.MethodInvokingJobDetailFactoryBean" >
          < property name= "targetObject" ref= "job" />
        < property name= "targetMethod" value= "runWork" />
        <!-- false 表示 job 不会并发执行,默认为 true-->
        < property name= "concurrent" value= "false" />
    </ bean >
     targetObject 属性指定的任务类,有多种方式实现。
    1 可以用 @Component 注解在类上面标注,这样就不用定义 < bean id= "job" ... /> 这些东西了。
    2 可以按上面的写法来配置。
    3 直接使用下面的写法。
    < property name= "targetObject" >
        < bean class= "xx.xx.xx.Job" />
    </ property >
     接下来配置触发器
    < bean id= "doWork" class= "org.springframework.scheduling.quartz.CronTriggerBean" >
        < property name= "jobDetail" ref= "cronTask" />
        <!— 每天凌晨 0 1 分执行 -->
        < property name= "cronExpression" value= "0 01 00 * * ?" />
    </ bean >
     最后配置调度工厂
    < bean class= "org.springframework.scheduling.quartz.SchedulerFactoryBean" >
        < property name= "triggers" >
            < list >
                < ref local= "doWork" />
            </ list >
        </ property >
     </ bean >
     到此,整个配置就完成了。下面再看看 Spring-Task 实现定时任务的步骤。
     Spring 3.0 开始增加了自己的任务调度器,它是通过扩展 java.util.concurrent 包下面的类来实现的,它也使用 Cron 表达式。
     使用 spring task 非常简单,首先增加命名空间 schema
     < beans xmlns= "http://www.springframework.org/schema/beans" 
         ......
          xmlns:task= "http://www.springframework.org/schema/task"
            xsi:schemaLocation= "
              ......
          http://www.springframework.org/schema/taskhttp://www.springframework.org/schema/task/spring-task-3.0.xsd" >
     然后给定时任务类添加 @Component 注解,给任务方法添加 @Scheduled(cron= "0/5 * * * * ?") 注解,并让 Spring 扫描到该类。
     然后加上 < task:annotation-driven/> 这个配置,让 Spring 识别 @Scheduled 注解 ( org.springframework.scheduling.annotation.Scheduled )
     OK ,设置完成。如果还想扩展一下,改成下面这样:
     < task:executor id= "executor" pool-size= "5" />
     < task:scheduler id= "scheduler" pool-size= "5" />
     < task:annotation-driven executor= "executor" scheduler= "scheduler" />
     如果定时任务很多,可以配置 executor 线程池,这里 executor 的含义和 java.util. concurrent. Executor 是一样的, pool-size 的大小官方推荐为 5~10 scheduler的pool-size是ScheduledExecutorService 线程池,默认为 1 。假如我设置了 8 个任务,每个任务都是每 5 秒钟执行一次,把下面的代码再复制 7 份再改一改,看看打印结果。
     @Scheduled(cron = "0/5 * * * * ?")
     public void work1(){
          System. out .println(Thread. currentThread().getName()+ " "+ "work1: 5 秒执行一次 ");
    }
    
     定时任务执行了 3 次,我们可以看到,线程名称都是以 scheduler 为前缀,这是因为我们已经在 < task:scheduler id= "scheduler" pool-size= "5" /> 这段配置里定义了 id scheduler 的结果,它就是用来作为任务线程的前缀,再交给 executor 线程 池进行。
     3 次任务执行,因为我们设定的任务调度线程池大小为 5 ,所以,只有 5 个实例来处理这 8 个任务,从结果可以看出来,不是每次都会用上全部的 5 个实例。如果你系统中的定时任务过多,这个 pool-size 的大小就应该调大一点,方便之前定义的 executor 线程池来执行。
     本文为菠萝大象原创,如要转载请注明出处。 http://www.blogjava.net/bolo
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值