天天看點

Quartz核心知識核心概念cron 表達式的格式

上一篇我們寫一個關于quartz的demo,可以說是直接性開門見山感受quartz的功能,然而為了搞清楚其功能,我們還需從其本質說起。這篇文章主要講述了quartz的核心知識,“”Quartz cron ”表達式。

隻有你熟練掌握這些表達式之後,然後根據你的業務需求,直接使用即可,完全沒什麼難點。相信自己即可。學習每一個新技術或者知識點首先弄明白其工作原理,根據執行流程有目的有方向地學習,相信會事半功倍。

核心概念

1、scheduler是一個計劃排程器容器,容器裡面可以盛放衆多的JobDetail和trigger,當容器啟動後,裡面的每個JobDetail都會根據trigger按部就班自動去執行。

2、JobDetail是一個可執行的工作,它本身可能是有狀态的。

3、Trigger觸發器代表一個排程參數的配置,什麼時候去調。

4、當JobDetail和Trigger在scheduler容器上注冊後,形成了裝配好的作業(JobDetail和Trigger所組成的一對兒),就可以伴随容器啟動而排程執行了。

5、scheduler是個容器,容器中有一個線程池,用來并行排程執行每個作業,這樣可以提高容器效率。

cron 表達式的格式

Quartz cron 表達式的格式十分類似于 UNIX cron 格式,但還是有少許明顯的差別。差別之一就是 Quartz 的格式向下支援到秒級别的計劃,而 UNIX cron 計劃僅支援至分鐘級。許多我們的觸發計劃要基于秒級遞增的(例如,每45秒),是以這是一個非常好的差異。

在 UNIX cron 裡,要執行的作業(或者說指令)是存放在 cron 表達式中的,在第六個域位置上。Quartz 用 cron 表達式存放執行計劃。引用了 cron 表達式的 CronTrigger 在計劃的時間裡會與 job 關聯上。

另一個與 UNIX cron 表達式的不同點是在表達式中支援域的數目。UNIX 給出五個域(分、時、日、月和周),Quartz 提供七個域。表 5.1 列出了 Quartz cron 表達式支援的七個域。

Quartz Cron 表達式支援到七個域 

名稱 是否必須 允許值 特殊字元
0-59 , - * /
0-59 , - * /
0-23 , - * /
1-31 , - * ? / L W C
1-12 或 JAN-DEC , - * /
1-7 或 SUN-SAT , - * ? / L C #
空 或 1970-2099 , - * /

月份和星期的名稱是不區分大小寫的。FRI 和 fri 是一樣的。 

域之間有空格分隔,這和 UNIX cron 一樣。無可争辯的,我們能寫的最簡單的表達式看起來就是這個了: 

* * * ? * * 

這個表達會每秒鐘(每分種的、每小時的、每天的)激發一個部署的 job。 

·了解特殊字元 

同 UNIX cron 一樣,Quartz cron 表達式支援用特殊字元來建立更為複雜的執行計劃。然而,Quartz 在特殊字元的支援上比标準 UNIX cron 表達式更豐富了。 

* 星号 

使用星号(*) 訓示着你想在這個域上包含所有合法的值。例如,在月份域上使用星号意味着每個月都會觸發這個 trigger。

表達式樣例: 

0 * 17 * * ? 

意義:每天從下午5點到下午5:59中的每分鐘激發一次 trigger。它停在下午 5:59 是因為值 17 在小時域上,在下午 6 點時,小時變為 18 了,也就不再理會這個 trigger,直到下一天的下午5點。 

在你希望 trigger 在該域的所有有效值上被激發時使用 * 字元。 

? 問号 

? 号隻能用在日和周域上,但是不能在這兩個域上同時使用。你可以認為 ? 字元是 "我并不關心在該域上是什麼值。" 這不同于星号,星号是訓示着該域上的每一個值。? 是說不為該域指定值。 

不能同時這兩個域上指定值的理由是難以解釋甚至是難以了解的。基本上,假定同時指定值的話,意義就會變得含混不清了:考慮一下,如果一個表達式在日域上有值11,同時在周域上指定了 WED。那麼是要 trigger 僅在每個月的11号,且正好又是星期三那天被激發?還是在每個星期三的11号被激發呢?要去除這種不明确性的辦法就是不能同時在這兩個域上指定值。 

隻要記住,假如你為這兩域的其中一個指定了值,那就必須在另一個字值上放一個 ?。 

表達式樣例: 

0 10,44 14 ? 3 WEB 

意義:在三月中的每個星期三的下午 2:10 和 下午 2:44 被觸發。 

, 逗号 

逗号 (,) 是用來在給某個域上指定一個值清單的。例如,使用值 0,15,30,45 在秒域上意味着每15秒觸發一個 trigger。 

表達式樣例: 

0 0,15,30,45 * * * ? 

意義:每刻鐘觸發一次 trigger。 

/ 斜杠 

斜杠 (/) 是用于時間表的遞增的。我們剛剛用了逗号來表示每15分鐘的遞增,但是我們也能寫成這樣 0/15。 

表達式樣例: 

0/15 0/30 * * * ? 

意義:在整點和半點時每15秒觸發 trigger。 

- 中劃線 

中劃線 (-) 用于指定一個範圍。例如,在小時域上的 3-8 意味着 "3,4,5,6,7 和 8 點。"  域的值不允許回卷,是以像 50-10 這樣的值是不允許的。 

表達式樣例: 

0 45 3-8 ? * * 

意義:在上午的3點至上午的8點的45分時觸發 trigger。 

L 字母

L 說明了某域上允許的最後一個值。它僅被日和周域支援。當用在日域上,表示的是在月域上指定的月份的最後一天。例如,當月域上指定了 JAN 時,在日域上的 L 會促使 trigger 在1月31号被觸發。假如月域上是 SEP,那麼 L 會預示着在9月30号觸發。換句話說,就是不管指定了哪個月,都是在相應月份的時最後一天觸發 trigger。 

表達式 0 0 8 L * ? 意義是在每個月最後一天的上午 8:00 觸發 trigger。在月域上的 * 說明是 "每個月"。 

當 L 字母用于周域上,訓示着周的最後一天,就是星期六 (或者數字7)。是以如果你需要在每個月的最後一個星期六下午的 11:59 觸發 trigger,你可以用這樣的表達式 0 59 23 ? * L。 

當使用于周域上,你可以用一個數字與 L 連起來表示月份的最後一個星期 X。例如,表達式 0 0 12 ? * 2L 說的是在每個月的最後一個星期一觸發 trigger。 

不要讓範圍和清單值與 L 連用

雖然你能用星期數(1-7)與 L 連用,但是不允許你用一個範圍值和清單值與 L 連用。這會産生不可預知的結果。

W 字母 

W 字元代表着平日 (Mon-Fri),并且僅能用于日域中。它用來指定離指定日的最近的一個平日。大部分的商業處理都是基于工作周的,是以 W 字元可能是非常重要的。例如,日域中的 15W 意味着 "離該月15号的最近一個平日。" 假如15号是星期六,那麼 trigger 會在14号(星期四)觸發,因為距15号最近的是星期一,這個例子中也會是17号(譯者Unmi注:不會在17号觸發的,如果是15W,可能會是在14号(15号是星期六)或者15号(15号是星期天)觸發,也就是隻能出現在鄰近的一天,如果15号當天為平日直接就會當日執行)。W 隻能用在指定的日域為單天,不能是範圍或清單值。 

# 井号 

# 字元僅能用于周域中。它用于指定月份中的第幾周的哪一天。例如,如果你指定周域的值為 6#3,它意思是某月的第三個周五 (6=星期五,#3意味着月份中的第三周)。另一個例子 2#1 意思是某月的第一個星期一 (2=星期一,#1意味着月份中的第一周)。注意,假如你指定 #5,然而月份中沒有第 5 周,那麼該月不會觸發。

示例:

表達式意義

"0 0 12 * *?" 每天中午12點觸發

"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 ? 3WED" 每年三月的星期三的下午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 ? * 6L2002-2005" 2002年至2005年的每月的最後一個星期五上午10:15觸發

"0 15 10 ? *6#3" 每月的第三個星期五上午10:15觸發

繼續閱讀