上一文中提到 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表示一個月的第三個星期五