java quartz 中的时间格式

 

SimpleTrigger
如果需要计划一个任务在指定的时间执行,或者在指定的时间后以指定的间隔连续执行多次,比如希望在2005年1月12号上午11:22:54开始执行一个任务,在这之后每隔20分钟执行一次,共执行一次,这种情况下可以使用SimpleTrigger。

SimpleTrigger包含几个属性:开始时间,结束时间,重复次数和间隔。

重复次数可以是大于等于0,或者是常量值SimpleTrigger.REPEAT_INDEFINITELY,间隔必须大于等于0的长整数,单位是微秒。如果间隔为0表示并发执行重复次数。

如果不熟悉java.util.Calendar类,可能经常需要根据开始时间计算触发时间,org.quartz.helpers.TriggerUtils 可以帮助完成这些任务。

结束时间属性重写重复次数属性。如果希望创建一个触发器,每隔10秒执行一次,直到一个指定的时间,可以简单的指定结束时间, 重复次数值为REPEAT_INDEFINITELY。

SimpleTrigger有几个构造函数,下面是其中一个:

java 代码
public SimpleTrigger(String name,    
                       String group,    
                       Date startTime,    
                       Date endTime,    
                       int repeatCount,    
                       long repeatInterval)    
创建一个10秒钟后只执行一次的触发器:

java 代码
long startTime = System.currentTimeMillis() + 10000L;    
   
SimpleTrigger trigger = new SimpleTrigger("myTrigger",    
                                            null,    
                                            new Date(startTime),    
                                            null,    
                                            0,    
                                            0L);    
创建一个每隔60秒重复执行的触发器:

SimpleTrigger trigger = new SimpleTrigger("myTrigger",    
                                            null,    
                                            new Date(),    
                                            null,    
                                            SimpleTrigger.REPEAT_INDEFINITELY,    
                                            60L * 1000L);   
java 代码
创建一个40秒后开始执行,每隔10秒执行一次的触发器:

java 代码
long endTime = System.currentTimeMillis() + 40000L;    
   
SimpleTrigger trigger = new SimpleTrigger("myTrigger",    
                                            "myGroup",    
                                            new Date(),    
                                            new Date(endTime),    
                                            SimpleTrigger.REPEAT_INDEFINITELY,    
                                            10L * 1000L);    
创建一个触发器,在2002年3月17日开始执行,重复5次,每次间隔为30秒:

java 代码
 

java.util.Calendar cal = new java.util.GregorianCalendar(2002, cal.MARCH, 17);    
  cal.set(cal.HOUR, 10);    
  cal.set(cal.MINUTE, 30);    
  cal.set(cal.SECOND, 0);    
  cal.set(cal.MILLISECOND, 0);  Data startTime = cal.getTime()  SimpleTrigger trigger = new SimpleTrigger("myTrigger",    
                                            null,    
                                            startTime,    
                                            null,    
                                            5,    
                                            30L * 1000L);   
CronTrigger

如果需要基于日历指定触发器,可以使用CronTrigger。使用CronTrigger可以实现类似的触发器,比如:每个星期五的下午。比如每个星期一,三和五的上午9点到10点之间每隔5分钟。

CronTrigger也有一个开始时间和结束时间属性,用来指定什么时候任务开始和结束。

Cron表达式
*Cron*表达式用来配置CronTrigger。Cron表达式是一个由七个部分组成的字符串,这七个部分用空隔进行分隔:

Seconds 
Minutes 
Hours 
Day-of-Month 
Month 
Day-of-Week 
Year (可选字段) 
'*'字符表示指定字段的所有可能值,比如Day-Of-Week字段的*表示每天。

每个字段都有一些有效值。比如秒和分可以取值0-59,小时可以取值0-23。Day-of-Month可以取值0-31,需要注意一个月有多少天。 月可以取值0-11,或者通过使用JAN, FEB, MAR, APR, MAY, JUN, JUL, AUG, SEP, OCT, NOV 和 DEC。 Days-of-Week可以取值1-7(1==Sunday)或者SUN, MON, TUE, WED, THU, FRI 和SAT。

'/' 字符可以用来指定增量,比如如果指定Minute字段为"0/15"表示在第0分钟启动,每隔15分钟的间隔;"3/20"表示每三分钟启动,每隔20分钟的间隔。

'?' 字符可以在day-of-month和day-of-week 字段中使用,用来表示没有指定值。

'L'字符可以在day-of-month和day-of-week 字段中使用,这个字符表示最后一个的意思。比如在day-of-month字段中表示这个月的最后一天,如果在day-of-week字段表示"7"或者"SAT",但是如果在day-of-week字段L在另一个值后面,意味着这个月的最后XXX天,比如"6L"表示这个月的最后一个星期五。使用这个字符,不能指定列表,范围值。

'W'字符用来指定离指定天最*的星期XXX,比如如果day-of-month字段值为"15W",表示离这个月15号最*的一个weekday。

'#'字符用来表示这个月的第几个XXX,比如day-of-week字段的"6#3"表示这个月的第三个星期五。

下面是一些示例:
创建一个每五分钟激活一次的触发器:

java 代码
"0 0/5 * * * ?"    
创建一个触发器在当前分钟的第10秒后,每五分钟执行一次,比如上午10:00:10 am,上午10:05:10:

java 代码
"10 0/5 * * * ?"    
创建一个触发器,在每个星期三和星期五的10:30, 11:30, 12:30, 和13:30执行。

java 代码
"0 30 10-13 ? * WED,FRI"    
创建一个触发器,在每个月的第5天和第20天的上午8点到10点执行,每隔半小时执行一次,注意上午10:00不会执行:

java 代码
"0 0/30 8-9 5,20 * ?"  

------------------------------------------------------------------

 

上一文中提到 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

 

posted @ 2018-08-13 22:30  roychenyi  阅读(2133)  评论(0编辑  收藏  举报