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線程池來執行。

相關文章
相關標籤/搜索