天天看點

Quartz Cron 觸發器 Cron Expression 的格式 Quartz Cron Expression

上一文中提到 Cron觸發器可以接受一個表達式來指定執行JOB,下面看看這個表達式的文法。

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 表達式支援的七個域。

表 5.1. 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 周,那麼該月不會觸發。

Cron 表達式 Cookbook

此處的 Cron 表達式 cookbook 旨在為常用的執行需求提供方案。盡管不可能列舉出所有的表達式,但下面的應該為滿足你的業務需求提供了足夠的例子。

·分鐘的 Cron 表達式

表 5.1. 包括了分鐘頻度的任務計劃 Cron 表達式 用法 表達式

每天的從 5:00 PM 至 5:59 PM 中的每分鐘觸發 0 * 17 * * ?

每天的從 11:00 PM 至 11:55 PM 中的每五分鐘觸發 0 0/5 23 * * ?

每天的從 3:00 至 3:55 PM 和 6:00 PM 至 6:55 PM 之中的每五分鐘觸發 0 0/5 15,18 * * ?

每天的從 5:00 AM 至 5:05 AM 中的每分鐘觸發 0 0-5 5 * * ?

·日上的 Cron 表達式

表 5.2. 基于日的頻度上任務計劃的 Cron 表達式 用法 表達式

每天的 3:00 AM  0 0 3 * * ?

每天的 3:00 AM (另一種寫法)  0 0 3 ? * *

每天的 12:00 PM (中午) 0 0 12 * * ?

在 2005 中每天的 10:15 AM 0 15 10 * * ? 2005

·周和月的 Cron 表達式

表 5.3. 基于周和/或月的頻度上任務計劃的 Cron 表達式 用法 表達式

在每個周一,二, 三和周四的 10:15 AM 0 15 10 ? * MON-FRI

每月15号的 10:15 AM   0 15 10 15 * ?

每月最後一天的 10:15 AM 0 15 10 L * ?

每月最後一個周五的 10:15 AM  0 15 10 ? * 6L

在 2002, 2003, 2004, 和 2005 年中的每月最後一個周五的 10:15 AM  0 15 10 ? * 6L 2002-2005

每月第三個周五的 10:15 AM 0 15 10 ? * 6#3

每月從第一天算起每五天的 12:00 PM (中午) 0 0 12 1/5 * ?

每一個 11 月 11 号的 11:11 AM 0 11 11 11 11 ?

三月份每個周三的 2:10 PM 和 2:44 PM 0 10,44 14 ? 3 WED

Quartz Cron Expression

Field Name Mandatory? Allowed Values Allowed Special Characters
Seconds YES 0-59 , - * /
Minutes YES 0-59 , - * /
Hours YES 0-23 , - * /
Day of month YES 1-31 , - * ? / L W C
Month YES 1-12 or JAN-DEC , - * /
Day of week YES 1-7 or SUN-SAT , - * ? / L C #
Year NO empty, 1970-2099 , - * /

  項目執行個體:               second  minute  hours  dayOfMonth  month  dayOfWeek  year 每 月         0            0           6              ?                    *                6#3            ? 每 周        59           59         18            ?                    *                1                ? 自定義    28          47          9             30                 7                ?             2006   每 月:每個月的第三個星期五的上午 6:00:00 觸發 每 周:每周的星期日的下午18:59:59 觸發 自 定義:2006年7月30日上午 9:47:28 觸發  

所有星号對應的段位置,都可以出現後面的符号(, - * /)

(? / L C)這些符号可以出現在"一月哪天"和"星期"段位置

(w)隻能出現在"一月哪天"段位置

(#)隻能出現在"星期"段位置

解釋符号代表的意思:

* 代表任意合法的字段

0 * 17 * * ? :表示在每天的5 PM 到 5:59之間的每一分鐘啟動scheduler

? 表示沒值被指定

如果同時指定"一月哪天"和"星期",可能兩者對應不起來

0 0,15,30,45 * * * ? :表示每刻鐘啟動scheduler

是以推薦用法是其中一個指定值,另一個用?指定

/ 表示時間的增量

0 0/15 * * * ? :表示每刻鐘啟動scheduler

- 表示值的範圍

0 45 3-8 ? * *

L 如果用在"一月哪天"段上,表示一個月的最後一天;如果用在"星期"段上。表示一個星期的最後一天(星期六)

0 0 8 L * ? :表示每個月最後一天的8點啟動scheduler

W 表示最靠近給定時間的一天,(必須是星期一到星期五)

# 例如 6#3表示一個月的第三個星期五