對于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"放在周內日期字段中,表示把任務安排在當月的第二個星期一。
星號(*)字符是通配字符,表示該字段可以接受任何可能的值、表達式例子。
例子:
"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:10和2: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" />
<!—每天凌晨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/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~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.aygfsteel.com/bolo