常用的Cron Schedule
相对于其他方式定义定时任务的触发时间,我们较常用Cron Schedule,大概是触发时间点的配置看起来比较简洁吧
Cron Schedule的使用
// 具体任务 JobDetail job = JobBuilder.newJob(HelloJob.class).withIdentity("job1", "group1").build(); // 触发时间点 CronScheduleBuilder cronScheduleBuilder = CronScheduleBuilder.cronSchedule("0 * * * * ? *"); // 触发器配置 Trigger trigger = TriggerBuilder.newTrigger().withIdentity("trigger1", "group1").withSchedule(cronScheduleBuilder).build(); // 交由Scheduler安排触发 scheduler.scheduleJob(job, trigger);
Cron Schedule用的了Cron表达式。【秒】【分】【时】【日】【月】【周】【年】
在线生成Cron表达式的网站特别多:http://cron.qqe2.com、http://www.pdtools.net/tools/becron.jsp
cron表达式中的问号(?) 是什么意思?
问号(?)的作用是指明该字段‘没有特定的值’
其实这也很好理解,比如:"0/5 * * 3 * SUN",既指定了每个月的3号,并且又指定了每个星期天,那么并不可能存在每个3号都是星期天的可能性,所以,表达式也就不成立了。
cronExpression对日期和星期字段的处理规则是它们必须互斥,即只能且必须有一个字段有特定的值,另一个字段必须是‘没有特定的值’;
1、当星期和日期都为*或数字时,报错
Support for specifying both a day-of-week AND a day-of-month parameter is not implemented.
即两个字段不能都指明的特定的值,必须互斥。这里的*和数字是一样的,如果都指明特定的数字,也是报一样的错。
2、当星期和日期都为?时,报错
'?' can only be specfied for Day-of-Month -OR- Day-of-Week.
即两个字段不能都‘没有特定的值’。
时间格式
字段 | 允许的值 | 允许的特殊字符 |
Seconds 秒 |
0-59 |
, - * / |
Minutes 分 |
0-59 |
, - * / |
Hours 小时 |
0-23 |
, - * / |
Day-of-Month 日 |
1-31 |
, - * ? / L W C |
Month 月 |
0-11 或 JAN、FEB、MAR、APR、MAY、JUN、JUL、AUG、SEP、OCT、NOV、DEC |
, - * / |
Day-of-Week 周几 |
1-7 或 SUN、MON、TUE、WED、THU、FRI、SAT |
, - * ? / L C # |
Year 年 (可选字段) |
empty, 1970-2099 |
, - * / |
特殊字符
特殊字符 | 含义 |
* | 字符可以用于所有字段,在“分”字段中设为"*"表示"每一分钟"的含义 |
? | 字符可以用在“日”和“周几”字段. 它用来指定 '不明确的值'. 这在你需要指定这两个字段中的某一个值而不是另外一个的时候会被用到 |
- | 字符被用来指定一个值的范围,比如在“小时”字段中设为"10-12"表示"10点到12点" |
, | 字符指定数个值。比如在“周几”字段中设为"MON,WED,FRI"表示"the days Monday, Wednesday, and Friday" |
/ | 字符用来指定一个值的的增加幅度,比如在“秒”字段中设置为"0/15"表示"第0, 15, 30, 和 45秒"。而 "5/15"则表示"第5, 20, 35, 和 50". 在'/'前加"*"字符相当于指定从0秒开始. 每个字段都有一系列可以开始或结束的数值。对于“秒”和“分”字段来说,其数值范围为0到59,对于“小时”字段来说其为0到23, 对于“日”字段来说为0到31, 而对于“月”字段来说为0到11。"/"字段仅仅只是帮助你在允许的数值范围内从开始"第n"的值。 因此对于“月”字段来说"7/6"只是表示7月被开启而不是“每六个月”, 请注意其中微妙的差别 |
L | 字符可用在“日”和“周几”这两个字段。它是"last"的缩写, 但是在这两个字段中有不同的含义。例如,“日”字段中的"L"表示"一个月中的最后一天" —— 对于一月就是31号对于二月来说就是28号(非闰年)。而在“周几”字段中, 它简单的表示"7" or "SAT",但是如果在“周几”字段中使用时跟在某个数字之后, 它表示"该月最后一个星期×" —— 比如"6L"表示"该月最后一个周五"。当使用'L'选项时,指定确定的列表或者范围非常重要,否则你会被结果搞糊涂的 |
W | 可用于“日”字段。用来指定历给定日期最近的工作日(周一到周五) 。比如你将“日”字段设为"15W",意为: "离该月15号最近的工作日"。因此如果15号为周六,触发器会在14号即周五调用。如果15号为周日, 触发器会在16号也就是周一触发。如果15号为周二,那么当天就会触发。然而如果你将“日”字段设为"1W", 而一号又是周六, 触发器会于下周一也就是当月的3号触发,因为它不会越过当月的值的范围边界。'W'字符只能用于“日”字段的值为单独的一天而不是一系列值的时候 |
LW | 可以组合用于“日”字段表示为'LW',意为"该月最后一个工作日" |
# | 字符可用于“周几”字段。该字符表示“该月第几个周×”,比如"6#3"表示该月第三个周五( 6表示周五而"#3"该月第三个)。再比如: "2#1" = 表示该月第一个周一而 "4#5" = 该月第五个周三。注意如果你指定"#5"该月没有第五个“周×”,该月是不会触发的 |
C | 字符可用于“日”和“周几”字段,它是"calendar"的缩写。它表示为基于相关的日历所计算出的值(如果有的话)。如果没有关联的日历, 那它等同于包含全部日历。“日”字段值为"5C"表示"日历中的第一天或者5号以后",“周几”字段值为"1C"则表示"日历中的第一天或者周日以后" |
对于“月份”字段和“周几”字段来说合法的字符都不是大小写敏感的。
下面是一些完整的例子
表达式 | 含义 |
"0 0 12 * * ?" |
每天中午十二点触发 |
"0 15 10 ? * *" |
每天早上10:15触发 |
"0 15 10 * * ?" |
每天早上10:15触发 |
"0 15 10 * * ? *" |
每天早上10:15触发 |
"0 15 10 * * ? 2005" |
2005年的每天早上10:15触发 |
"0 * 14 * * ?" |
每天从下午2点开始到2点59分每分钟一次触发 |
"0 0/5 14 * * ?" |
每天从下午2点开始到2:55分结束每5分钟一次触发 |
"0 0/5 14,18 * * ?" |
每天的下午2点至2:55和6点至6点55分两个时间段内每5分钟一次触发 |
"0 0-5 14 * * ?" |
每天14:00至14:05每分钟一次触发 |
"0 10,44 14 ? 3 WED" |
三月的每周三的14:10和14:44触发 |
"0 15 10 ? * MON-FRI" |
每个周一、周二、周三、周四、周五的10:15触发 |
"0 15 10 15 * ?" |
每月15号的10:15触发 |
"0 15 10 L * ?" |
每月的最后一天的10:15触发 |
"0 15 10 ? * 6L" |
每月最后一个周五的10:15触发 |
"0 15 10 ? * 6L" |
每月最后一个周五的10:15触发 |
"0 15 10 ? * 6L 2002-2005" |
2002年至2005年的每月最后一个周五的10:15触发 |
"0 15 10 ? * 6#3" |
每月的第三个周五的10:15触发 |
"0 15 10 LW * ?" | 每个月最后一个工作日的10点15分0秒触发任务 |
"0 15 10 L * ?" | 每个月最后一天的10点15分0秒触发任务 |
"0 0/3 * * * ?" | 每小时的第0分0秒开始,每三分钟触发一次 |
"15-30/5 * * * * ?" | 每分钟的15秒到30秒之间开始触发,每隔5秒触发一次 |
"15/5 * * * * ?" | 每分钟的每15秒开始触发,每隔5秒触发一次 |
"15-45 * * * * ?" | 15到45秒内,每秒都触发任务 |
"15,30,45 * * * * ?" | 每15秒,30秒,45秒时触发任务 |
"30 * * * * ?" | 每半分钟触发任务 |
"30 10 * * * ?" | 每小时的10分30秒触发任务 |
"30 10 1 * * ?" | 每天1点10分30秒触发任务 |
"30 10 1 20 * ?" | 每月20号1点10分30秒触发任务 |
"30 10 1 20 10 ? *" | 每年10月20号1点10分30秒触发任务 |
"30 10 1 20 10 ? 2011" | 2011年10月20号1点10分30秒触发任务 |
"30 10 1 ? 10 * 2011" | 2011年10月每天1点10分30秒触发任务 |
"30 10 1 ? 10 SUN 2011" | 2011年10月每周日1点10分30秒触发任务 |
构建CronTriggers
CronTrigger实例使用TriggerBuilder(用于触发器的主要属性)和CronScheduleBuilder(对于CronTrigger特定的属性)构建。 要以DSL风格使用这些构建器,请使用静态导入:
import static org.quartz.TriggerBuilder.*; import static org.quartz.CronScheduleBuilder.*; import static org.quartz.DateBuilder.*:
建立一个触发器,每隔一分钟,每天上午8点至下午5点之间:
trigger = newTrigger() .withIdentity("trigger3", "group1") .withSchedule(cronSchedule("0 0/2 8-17 * * ?")) .forJob("myJob", "group1") .build();
建立一个触发器,将在上午10:42每天触发:
trigger = newTrigger() .withIdentity("trigger3", "group1") .withSchedule(dailyAtHourAndMinute(10, 42)) .forJob(myJobKey) .build();
或者
trigger = newTrigger() .withIdentity("trigger3", "group1") .withSchedule(cronSchedule("0 42 10 * * ?")) .forJob(myJobKey) .build();
建立一个触发器,将在星期三上午10:42在TimeZone(系统默认值)之外触发:
trigger = newTrigger() .withIdentity("trigger3", "group1") .withSchedule(weeklyOnDayAndHourAndMinute(DateBuilder.WEDNESDAY, 10, 42)) .forJob(myJobKey) .inTimeZone(TimeZone.getTimeZone("America/Los_Angeles")) .build();
或者
trigger = newTrigger() .withIdentity("trigger3", "group1") .withSchedule(cronSchedule("0 42 10 ? * WED")) .inTimeZone(TimeZone.getTimeZone("America/Los_Angeles")) .forJob(myJobKey) .build();
官网介绍:http://www.quartz-scheduler.org/documentation/quartz-2.2.x/tutorials/tutorial-lesson-06.html