1. 程式人生 > >Quartz Spring與Spring Task總結

Quartz Spring與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觸發     使用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" />        <!—每天凌晨01分執行-->        <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實現定時任務的步驟。     Spring3.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/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,設定完成。如果還想擴充套件一下,改成下面這樣:     <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份再改一改,看看列印結果。     @Scheduled(cron = "0/5 * * * * ?")     public void work1(){         System.out.println(Thread.currentThread().getName()+" "+"work1: 5秒執行一次");     }          定時任務執行了3次,我們可以看到,執行緒名稱都是以scheduler為字首,這是因為我們已經在<task:scheduler id="scheduler" pool-size="5" />這段配置裡定義了idscheduler的結果,它就是用來作為任務執行緒的字首,再交給executor執行緒池進行。     3次任務執行,因為我們設定的任務排程執行緒池大小為5,所以,只有5個例項來處理這8個任務,從結果可以看出來,不是每次都會用上全部的5個例項。如果你係統中的定時任務過多,這個pool-size的大小就應該調大一點,方便之前定義的executor執行緒池來執行。     本文為菠蘿大象原創,如要轉載請註明出處。