Spring Quartz与Spring Task总结

    SpringQuartz做了一个封装,同时,Spring本身也提供了一个任务定时器(spring-task),现把它总结一下。
    对于Quartz,咱们使用的时候主要是注重两个方面,一个是定时任务的业务,另外一个就是Cron表达式。定时任务跟具体的业务相关,这无需多说,这里只说明表达式含义及其写法。
    Cron表达式包括下面7个字段并区别顺序0-590-59小时0-23月内日期1-311-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"放在周内日期字段中表示把任务安排在当月的第二个星期一
    星号(*)字符是通配字符,表示该字段能够接受任何可能的值表达式例子。
    例子:
    "0 0 08 * * ?" 天天上午8点触发
    "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 ? 3 WED" 每一年三月的星期三的下午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 ? * 6L 2009-2019" 2009年至2019年的每个月的最后一个星期五上午10:15触发
    "0 15 10 ? * 6#3" 每个月的第三个星期五上午10:15触发java

    使用Spring Quartz实现Job任务有两种方式,一种是继承org.springframework.scheduling.quartz.QuartzJobBean,这个不推荐。另外一种不须要继承,只须要在配置文件中定义org.springframework.scheduling.quartz.MethodInvokingJobDetailFactoryBean,并指定它的targetObject属性为Job任务类,targetMethod属性为任务方法就能够了。
spring

    <bean id="dealAppointmentTestTask"
        class="org.springframework.scheduling.quartz.MethodInvokingJobDetailFactoryBean">
        <property name="targetObject">
            <bean class="com.nesec.task.SpringTaskController" />
        </property>
        <property name="targetMethod">
            <value>task1</value>
        </property>
        <property name="concurrent" value="false" />
    </bean>
<bean id="job" class=" xx.xx.xx.Job" />

targetObject属性指定的任务类,有多种方式实现。
    1能够用@Component注解在类上面标注,这样就不用定义<bean id="job" ... />这些东西了。
    2能够按上面的写法来配置。
    3直接使用下面的写法。
spa

    <property name="targetObject">
        <bean class="xx.xx.xx.Job" />
    </property>

    接下来配置触发器
线程

    <bean id="dealAppointmentTestTrigger" class="org.springframework.scheduling.quartz.CronTriggerBean">
        <property name="jobDetail">
            <ref bean="dealAppointmentTestTask" />
        </property>
         <!—每5秒执行-->
        <property name="cronExpression">
            <value>0/5 * * * * ?</value>
        </property>
    </bean>

    最后配置调度工厂
code

    <bean autowire="no"
        class="org.springframework.scheduling.quartz.SchedulerFactoryBean">
        <property name="triggers">
            <list>
                <ref bean="dealAppointmentTestTrigger" />
            </list>
        </property>
    </bean>

    到此,整个配置就完成了。下面再看看Spring-Task实现定时任务的步骤。
    Spring3.0开始增长了本身的任务调度器,它是经过扩展java.util.concurrent包下面的类来实现的,它也使用Cron表达式。
    使用spring task很是简单,首先增长命名空间schema
xml

    <beans xmlns="http://www.springframework.org/schema/beans" 
        ......
        xmlns:task="http://www.springframework.org/schema/task"
            xsi:schemaLocation="
            ......
        http://www.springframework.org/schema/task http://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,设置完成。若是还想扩展一下,改为下面这样:
get

    <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~10schedulerpool-sizeScheduledExecutorService线程池,默认为1。假如我设置了8个任务,每一个任务都是每5秒钟执行一次,把下面的代码再复制7份再改一改,看看打印结果。
io

    @Scheduled(cron = "0/5 * * * * ?")
    public void work1(){
        System.out.println(Thread.currentThread().getName()+" "+"work1: 每5秒执行一次");
    }

......ast

定时任务执行了3次,咱们能够看到,线程名称都是以scheduler为前缀,这是由于咱们已经在<task:scheduler id="scheduler" pool-size="5" />这段配置里定义了idscheduler的结果,它就是用来做为任务线程的前缀,再交给executor线程池进行。
3次任务执行,由于咱们设定的任务调度线程池大小为5,因此,只有5个实例来处理这8个任务,从结果能够看出来,不是每次都会用上所有的5个实例。若是你系统中的定时任务过多,这个pool-size的大小就应该调大一点,方便以前定义的executor线程池来执行。

相关文章
相关标签/搜索