Spring對(duì)Quartz作了一個(gè)封裝,同時(shí),Spring自己也提供了一個(gè)任務(wù)定時(shí)器(spring-task),現(xiàn)把它總結(jié)一下。
對(duì)于Quartz,我們使用的時(shí)候主要是注重兩個(gè)方面,一個(gè)是定時(shí)任務(wù)的業(yè)務(wù),另一個(gè)就是Cron表達(dá)式。定時(shí)任務(wù)跟具體的業(yè)務(wù)相關(guān),這無需多說,這里只說明表達(dá)式含義及其寫法。 Cron表達(dá)式包括下面7個(gè)字段并區(qū)別順序:秒0-59,分0-59,小時(shí)0-23,月內(nèi)日期1-31,月1-12或者JAN-DEC,周內(nèi)日期1-7或者SUN-SAT,年(可選字段)留空或者1970-2099并且通過特殊字符表示特殊意義,具體為下: 斜線(/)字符表示增量值。例如,在秒字段中"5/15"代表從第5秒開始,每15秒一次。
問號(hào)(?)字符和字母L字符只有在月內(nèi)日期和周內(nèi)日期字段中可用。問號(hào)表示這個(gè)字段不包含具體值。所以,如果指定月內(nèi)日期,可以在周內(nèi)日期字段中插入"?",表示周內(nèi)日期值無關(guān)緊要。這里有個(gè)很蛋疼的設(shè)定,無關(guān)Quartz,而是Spring集成Quartz后,它自己加的一個(gè)約束,那就是:日期(1-31)和星期(SUN-SAT)兩者,必須有一個(gè)是問號(hào)(?),系統(tǒng)在啟動(dòng)的時(shí)候,Spring會(huì)檢查表達(dá)式,如果不符合它的規(guī)則,就會(huì)拋異常。所以在使用的時(shí)候這個(gè)地方一定要注意,而這個(gè)在Linux上執(zhí)行Cron是沒有這個(gè)限制的。 字母L字符是last的縮寫。放在月內(nèi)日期字段中,表示安排在當(dāng)月最后一天執(zhí)行。在周內(nèi)日期字段中,如果"L"單獨(dú)存在,就等于"7",否則代表當(dāng)月內(nèi)周內(nèi)日期的最后一個(gè)實(shí)例。所以"0L"表示安排在當(dāng)月的最后一個(gè)星期日?qǐng)?zhí)行。
字母(W)字符把執(zhí)行安排在最靠近指定值的工作日。把"1W"放在月內(nèi)日期字段中,表示把執(zhí)行安排在當(dāng)月的第一個(gè)工作日內(nèi)。 井號(hào)(#)字符為給定月份指定具體的工作日實(shí)例。把"MON#2"放在周內(nèi)日期字段中,表示把任務(wù)安排在當(dāng)月的第二個(gè)星期一。 星號(hào)(*)字符是通配字符,表示該字段可以接受任何可能的值、表達(dá)式例子。 例子: "0
0 08 * * ?" 每天上午8點(diǎn)觸發(fā)
"0
15 10 ? * *" 每天上午10:15觸發(fā) "0
15 10 * * ?" 每天上午10:15觸發(fā) "0
15 10 * * ? *" 每天上午10:15觸發(fā) "0 15 10 * * ? 2005" 2005年的每天上午10:15觸發(fā) "0
* 14 * * ?" 在每天下午2點(diǎn)到下午2:59期間的每1分鐘觸發(fā) "0
0/5 14 * * ?" 在每天下午2點(diǎn)到下午2:55期間的每5分鐘觸發(fā) "0
0/5 14,18 * * ?" 在每天下午2點(diǎn)到2:55期間和下午6點(diǎn)到6:55期間的每5分鐘觸發(fā) "0
0-5 14 * * ?" 在每天下午2點(diǎn)到下午2:05期間的每1分鐘觸發(fā) "0
10,44 14 ? 3 WED" 每年三月的星期三的下午2:10和2:44觸發(fā) "0
15 10 ? * MON-FRI" 周一至周五的上午10:15觸發(fā) "0
15 10 15 * ?" 每月15日上午10:15觸發(fā) "0
15 10 L * ?" 每月最后一日的上午10:15觸發(fā) "0
15 10 ? * 6L" 每月的最后一個(gè)星期五上午10:15觸發(fā) "0
15 10 ? * 6L 2009-2019" 2009年至2019年的每月的最后一個(gè)星期五上午10:15觸發(fā) "0
15 10 ? * 6#3" 每月的第三個(gè)星期五上午10:15觸發(fā)
使用Spring Quartz實(shí)現(xiàn)Job任務(wù)有兩種方式,一種是繼承org.springframework.scheduling.quartz.QuartzJobBean,這個(gè)不推薦。另一種不需要繼承,只需要在配置文件中定義org.springframework.scheduling.quartz.MethodInvokingJobDetailFactoryBean,并指定它的targetObject屬性為Job任務(wù)類,targetMethod屬性為任務(wù)方法就可以了。
<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不會(huì)并發(fā)執(zhí)行,默認(rèn)為true--> <property name="concurrent" value="false" /> </bean> targetObject屬性指定的任務(wù)類,有多種方式實(shí)現(xiàn)。 1、可以用@Component注解在類上面標(biāo)注,這樣就不用定義<bean id="job" ... />這些東西了。 2、可以按上面的寫法來配置。 3、直接使用下面的寫法。 <property name="targetObject"> <bean class="xx.xx.xx.Job" /> </property> 接下來配置觸發(fā)器 <bean id="doWork" class="org.springframework.scheduling.quartz.CronTriggerBean"> <property name="jobDetail" ref="cronTask"
/> <!—每天凌晨0點(diǎn)1分執(zhí)行--> <property name="cronExpression" value="0 01 00 * * ?" /> </bean> 最后配置調(diào)度工廠 <bean class="org.springframework.scheduling.quartz.SchedulerFactoryBean"> <property name="triggers"> <list> <ref local="doWork"/> </list> </property> </bean> 到此,整個(gè)配置就完成了。下面再看看Spring-Task實(shí)現(xiàn)定時(shí)任務(wù)的步驟。 Spring從3.0開始增加了自己的任務(wù)調(diào)度器,它是通過擴(kuò)展java.util.concurrent包下面的類來實(shí)現(xiàn)的,它也使用Cron表達(dá)式。 使用spring task非常簡(jiǎn)單,首先增加命名空間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"> 然后給定時(shí)任務(wù)類添加@Component注解,給任務(wù)方法添加@Scheduled(cron
= "0/5 * * * * ?")注解,并讓Spring掃描到該類。 然后加上<task:annotation-driven
/>這個(gè)配置,讓Spring識(shí)別@Scheduled注解(org.springframework.scheduling.annotation.Scheduled)。 OK,設(shè)置完成。如果還想擴(kuò)展一下,改成下面這樣: <task:executor id="executor" pool-size="5" /> <task:scheduler id="scheduler" pool-size="5" /> <task:annotation-driven executor="executor" scheduler="scheduler" /> 如果定時(shí)任務(wù)很多,可以配置executor線程池,這里executor的含義和java.util.concurrent.Executor是一樣的,pool-size的大小官方推薦為5~10。scheduler的pool-size是ScheduledExecutorService線程池,默認(rèn)為1。假如我設(shè)置了8個(gè)任務(wù),每個(gè)任務(wù)都是每5秒鐘執(zhí)行一次,把下面的代碼再復(fù)制7份再改一改,看看打印結(jié)果。 @Scheduled(cron = "0/5 * * * * ?") public void work1(){ System.out.println(Thread.currentThread().getName()+" "+"work1: 每5秒執(zhí)行一次"); }
定時(shí)任務(wù)執(zhí)行了3次,我們可以看到,線程名稱都是以scheduler為前綴,這是因?yàn)槲覀円呀?jīng)在<task:scheduler id="scheduler" pool-size="5" />這段配置里定義了id為scheduler的結(jié)果,它就是用來作為任務(wù)線程的前綴,再交給executor線程池進(jìn)行。 3次任務(wù)執(zhí)行,因?yàn)槲覀冊(cè)O(shè)定的任務(wù)調(diào)度線程池大小為5,所以,只有5個(gè)實(shí)例來處理這8個(gè)任務(wù),從結(jié)果可以看出來,不是每次都會(huì)用上全部的5個(gè)實(shí)例。如果你系統(tǒng)中的定時(shí)任務(wù)過多,這個(gè)pool-size的大小就應(yīng)該調(diào)大一點(diǎn),方便之前定義的executor線程池來執(zhí)行。 本文為菠蘿大象原創(chuàng),如要轉(zhuǎn)載請(qǐng)注明出處。http://www.tkk7.com/bolo
posted on 2015-03-12 16:15
菠蘿大象 閱讀(38575)
評(píng)論(4) 編輯 收藏 所屬分類:
Spring3