三. 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 表达式支持的七个域。
名称 | 是否必须 | 允许值 | 特殊字符 |
秒 | 是 | 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号(星期五)触发,因为星期四比星期一(这个例子中是17号)离15号更近。(译者Unmi注:不会在17号触发的,如果是15W,可能会是在14号(15号是星期六)或者15号(15号是星期天)触发,也就是只能出现在邻近的一天,如果15号当天为平日直接就会当日执行)。W 只能用在指定的日域为单天,不能是范围或列表值。
# 井号
# 字符仅能用于周域中。它用于指定月份中的第几周的哪一天。例如,如果你指定周域的值为 6#3,它意思是某月的第三个周五 (6=星期五,#3意味着月份中的第三周)。另一个例子 2#1 意思是某月的第一个星期一 (2=星期一,#1意味着月份中的第一周)。注意,假如你指定 #5,然而月份中没有第 5 周,那么该月不会触发。
本文链接 https://yanbin.blog/quartz-job-scheduling-framework-5-2/, 来自 隔叶黄莺 Yanbin Blog
[版权声明] 本文采用 署名-非商业性使用-相同方式共享 4.0 国际 (CC BY-NC-SA 4.0) 进行许可。
想请教一下,怎么表示7点到18点,每隔3个小时触发一次呢?谢谢。
你可以列出来啊,也就是在小时那个位置写成 7,10,13,16
请教一下
1.如果我想从2008年开始每2年执行一次 那要怎么写 不用罗列的有没有比较简捷的写法?
2.如果我想做一个任务是从2008年8月8日8点开始每小时都执行一次的任务如何定义?
谢谢了
0 0 0 1 1 ? 2008/2
2008年开始每2年的1月1号0点0分0秒执行
你的第二个要求更明显是应该用 SimpleTrigger 来处理的,从什么时候开始隔多长时间执行一次,标准的 SimpleTrigger 需求。
何必抱着 CronTriger 免为其难呢!
谢谢提醒。呵呵,原来是这样。
@WIN_ANGEL
@nesta
能讲得更详细吗
@大胡子
写错了
@Yanbin
@魔域私服
不错
请教一下:
1.每 n 小时的 0mm:0ss
2.每 n 天的 0hh:0mm:0ss
3.每 n 周的 Monday,Friday 的0hh:0mm:0ss
corn表达式怎么写?
谢谢你了!
每 n 的写法在相应域上用 0/n,比如分钟上的 0/20,就是从0开始的每20钟执行一次,也就是 0,20,40,60 这几个分点上
感谢大大的翻译
小弟有个问题
CronTriger 有办法达到 每1.5执行一次吗?
感觉好像只能用 SimpleTrigger 才能做到
什么是每 1.5 ?
例如,日域中的15W意味着 "离该月15号的最近一个平日。" 假如15号是星期六,那么trigger会在14号(星期四)触发,因为距15号最近的是星期一,这个例子中也会是17号
----------------------------------------------------------------
15号周6,14号应是周5;
W 字母这段没看懂。
For example, a value of 15W in the dayofmonth field means "the nearest weekday to the 15th of the month." If the 15th was on a Saturday, the trigger would fire on Friday the 14th because it's closer to the 15th than Monday, which would be the 17th in this example.
15W表示离这个月15号最近的工作日。如果15号是星期六,Trigger将在14号(星期五)执行。因为星期四比星期一(这个例子中是17号)离15号更近。
这个地方翻译得有点小错误。作者的意思是会在14号执行,而不是17号,因为14号离15号更近
受益匪浅啊
Unmi 帮我看看 September 1st 2010 at 15:08 这次留言!
呵呵,没有关系,翻译那么多内容难免出错!
我看到
第七章. 实现 Quartz 监听器 (第一部分)
对Quartz框架中如果用数据库存储JOB信息还是很不理解。
我有这样一种需求不知道能不能满足:
我制定一个任务的时候有2种方式,一种是立刻执行,第二种是预约执行就是在一个未来的时间点执行。
不知道用这个框架能否很好的实现,关键问题在于2个,首先是我在处理立即执行的任务时怎么去往数据库里面添加这样的一条信息;
这个时间点怎么去制定,太短了害怕当Quartz调度的时候这个时间点已经过去了,太远了害怕给客户的感觉响应很慢;
还有就是当一个任务执行完毕之后有没有状态标识他,或者将这个任务的信息给删除,让Quartz不去再执行一次。
当程序异常中断的时候,某些任务可能错过了触发时间,如何让这些任务重新执行?
是否可以留下QQ或者Email想和你交流一下。
Job 的数据库存储我用得也较少,如果选择了数据库存储则在执行时会产生一条记录,任务执行完后,好像它会把那条记录删除掉。任务的启动和结束可以用监听器监听得到的。
另外所谓的任务时间点,应该是一个时间频度,可以设定在某个时间点开始的每多少秒、分钟或每多少小时执行一次,这个起始的参照时间点可以是过去的某个时间,所以我想应该不会让客户感觉响应慢。
有些需求时可以使用 Job 链接的方式,即上一次 Job 执行完后由它来安排下一次 Job 的执行任务。错过的任务,如果超期时间不大于设置的 org.quartz.jobStore.misfireThreshold.
我不知你是不是用 Cron Expression 来安排 Job 的执行。
这个问题,好像以前有人问过的,晚上回去后我仔细看看。
”假如15号是星期六,那么 trigger 会在14号(星期四)触发,因为距15号最近的是星期一,这个例子中也会是17号“ 这句话看不懂,
应该是"假如15号是星期六,那么trigger会在14号(星期五)触发,因为距离15号最近的是周四。"
您书中的周是这样的 周日 周一 周二 周三 周四 周五 周六
还是楼主书看得仔细,不好意思,我对原文的理解以及当初的翻译欠准:
例如,日域中的 15W 意味着 “离该月15号的最近一个平日。” 假如15号是星期六,那么 trigger 会在14号(星期四)触发,因为距15号最近的是星期一,这个例子中也会是17号
这段话应该是:
例如,日域中的 15W 意味着 “离该月15号的最近一个平日。” 假如15号是星期六,那么 trigger 会在14号(星期四)触发,因为星期四比星期一(这个例子中是17号)离15号更近。
不知道这样的说法是否对楼上的理解有帮助。不管每周的起始日是周日还是周一,周一到周五就是平日,而周六周日是周末。
还是不对啊!
-------------------------
例如,日域中的 15W 意味着 “离该月15号的最近一个平日。” 假如15号是星期六,那么 trigger 会在14号(星期四)触发,因为星期四比星期一(这个例子中是17号)离15号更近。
-------------------------
15号是周六,那么14号是周五啊,要么是周日啊!怎么会是周四呢?
抱歉:
的确应该是:
例如,日域中的 15W 意味着 "离该月15号的最近一个平日。" 假如15号是星期六,那么 trigger 会在14号(星期五)触发,因为星期四比星期一(这个例子中是17号)离15号更近。