詳解定時任務中的 cron 表達式
1.前言
我們經常使用 cron 表達式來定義定時任務的執行策略,今天我們就總結一下 cron 表達式的一些相關知識。
2. cron 表達式的定義
cron 表達式是一個字符串,該字符串由 6
個空格分為 7
個域,每一個域代表一個時間含義。 格式如下:
[秒] [分] [時] [日] [月] [周] [年]
通常定義 “年” 的部分可以省略,實際常用的由 前六部分組成
2.1 cron各部定義
關於 cron 的各個域的定義如下表格所示:
域 | 是否必填 | 值以及范圍 | 通配符 |
---|---|---|---|
秒 | 是 | 0-59 | , - * / |
分 | 是 | 0-59 | , - * / |
時 | 是 | 0-23 | , - * / |
日 | 是 | 1-31 | , - * ? / L W |
月 | 是 | 1-12 或 JAN-DEC | , - * / |
周 | 是 | 1-7 或 SUN-SAT | , - * ? / L # |
年 | 否 | 1970-2099 | , - * / |
上面列表中值范圍還是比較好理解的,但是比較令開發者難以理解的就是通配符,其實 cron 表達式的難點也在於通配符。我們在下一個章節進行說明
2.2 cron中的通配符
,
這里指的是在兩個以上的時間點中都執行,如果我們在 “分” 這個域中定義為8,12,35
,則表示分別在第8分,第12分 第35分執行該定時任務。-
這個比較好理解就是指定在某個域的連續范圍,如果我們在 “時” 這個域中定義1-6
,則表示在1到6點之間每小時都觸發一次,用,
表示1,2,3,4,5,6
*
表示所有值,可解讀為 “每”。 如果在“日”這個域中設置*
,表示每一天都會觸發。?
表示不指定值。使用的場景為不需要關心當前設置這個字段的值。例如:要在每月的8號觸發一個操作,但不關心是周幾,我們可以這么設置0 0 0 8 * ?
/
在某個域上周期性觸發,該符號將其所在域中的表達式分為兩個部分,其中第一部分是起始值,除了秒以外都會降低一個單位,比如 在 “秒” 上定義5/10
表示從 第 5 秒開始 每 10 秒執行一次,而在 “分” 上則表示從 第 5 秒開始 每 10 分鍾執行一次。L
表示英文中的LAST 的意思,只能在 “日”和“周”中使用。在“日”中設置,表示當月的最后一天(依據當前月份,如果是二月還會依據是否是潤年), 在“周”上表示周六,相當於”7”或”SAT”。如果在”L”前加上數字,則表示該數據的最后一個。例如在“周”上設置”7L”這樣的格式,則表示“本月最后一個周六”W
表示離指定日期的最近那個工作日(周一至周五)觸發,只能在 “日” 中使用且只能用在具體的數字之后。若在“日”上置”15W”,表示離每月15號最近的那個工作日觸發。假如15號正好是周六,則找最近的周五(14號)觸發, 如果15號是周未,則找最近的下周一(16號)觸發.如果15號正好在工作日(周一至周五),則就在該天觸發。如果是 “1W” 就只能往本月的下一個最近的工作日推不能跨月往上一個月推。#
表示每月的第幾個周幾,只能作用於 “周” 上。例如 ”2#3” 表示在每月的第三個周二。
3. 示例
下面給出一些示例,可根據上面的解釋進行練習解讀:
- 每隔1分鍾執行一次:
0 */1 * * * ?
- 每天22點執行一次:
0 0 22 * * ?
- 每月1號凌晨1點執行一次:
0 0 1 1 * ?
- 每月最后一天23點執行一次:
0 0 23 L * ?
- 每周周六凌晨3點實行一次:
0 0 3 ? * L
- 在24分、30分執行一次:
0 24,30 * * * ?
4. 總結
今天總結了 cron 表達式,希望對你日常開發有所幫助。其實我們還可以借助於一些可視化的工具來生成 cron 表達式,但是如果我們同樣需要能夠看得懂別人寫的表達式。