Quartz定時任務框架(三)SimpleTrigger和CronTrigger、最詳細的Cron表示式範例
Quartz的主要介面類是Schedule、Job、Trigger,而觸發器Trigger就是定時任務的觸發時間,它規定安排了關聯的任務會在什麼時候執行,並且表明了再次執行的時機。
目錄
Trigger
在開始詳解每一種Trigger之前,需要先了解一下Trigger的一些共性。
StartTime & EndTime
startTime和endTime指定的Trigger會被觸發的時間區間。在這個區間之外,Trigger是不會被觸發的。
** 所有Trigger都會包含這兩個屬性 **
優先順序(Priority)
當scheduler比較繁忙的時候,可能在同一個時刻,有多個Trigger被觸發了,但資源不足(比如執行緒池不足)。那麼這個時候比剪刀石頭布更好的方式,就是設定優先順序。優先順序高的先執行。
需要注意的是,優先順序只有在同一時刻執行的Trigger之間才會起作用,如果一個Trigger是9:00,另一個Trigger是9:30。那麼無論後一個優先順序多高,前一個都是先執行。
優先順序的值預設是5,當為負數時使用預設值。最大值似乎沒有指定,但建議遵循Java的標準,使用1-10,不然鬼才知道看到【優先順序為10】是時,上頭還有沒有更大的值。
Misfire(錯失觸發)策略
類似的Scheduler資源不足的時候,或者機器崩潰重啟等,有可能某一些Trigger在應該觸發的時間點沒有被觸發,也就是Miss Fire了。這個時候Trigger需要一個策略來處理這種情況。每種Trigger可選的策略各不相同。
這裡有兩個點需要重點注意:
- MisFire的觸發是有一個閥值,這個閥值是配置在JobStore的。比RAMJobStore是org.quartz.jobStore.misfireThreshold。只有超過這個閥值,才會算MisFire。小於這個閥值,Quartz是會全部重新觸發。
所有MisFire的策略實際上都是解答兩個問題:
- 已經MisFire的任務還要重新觸發嗎?
- 如果發生MisFire,要調整現有的排程時間嗎?
SimpleTrigger的MisFire策略有:
-
MISFIRE_INSTRUCTION_IGNORE_MISFIRE_POLICY
這個不是忽略已經錯失的觸發的意思,而是說忽略MisFire策略。它會在資源合適的時候,重新觸發所有的MisFire任務,並且不會影響現有的排程時間。
比如,SimpleTrigger每15秒執行一次,而中間有5分鐘時間它都MisFire了,一共錯失了20個,5分鐘後,假設資源充足了,並且任務允許併發,它會被一次性觸發。
這個屬性是所有Trigger都適用。
-
MISFIRE_INSTRUCTION_FIRE_NOW
忽略已經MisFire的任務,並且立即執行排程。這通常只適用於只執行一次的任務。
-
MISFIRE_INSTRUCTION_RESCHEDULE_NOW_WITH_EXISTING_REPEAT_COUNT
將startTime設定當前時間,立即重新排程任務,包括的MisFire的
-
MISFIRE_INSTRUCTION_RESCHEDULE_NOW_WITH_REMAINING_REPEAT_COUNT
類似MISFIRE_INSTRUCTION_RESCHEDULE_NOW_WITH_EXISTING_REPEAT_COUNT,區別在於會忽略已經MisFire的任務
-
MISFIRE_INSTRUCTION_RESCHEDULE_NEXT_WITH_EXISTING_COUNT
在下一次排程時間點,重新開始排程任務,包括的MisFire的
-
MISFIRE_INSTRUCTION_RESCHEDULE_NEXT_WITH_REMAINING_COUNT
類似於MISFIRE_INSTRUCTION_RESCHEDULE_NEXT_WITH_EXISTING_COUNT,區別在於會忽略已經MisFire的任務。
-
MISFIRE_INSTRUCTION_SMART_POLICY
所有的Trigger的MisFire預設值都是這個,大致意思是“把處理邏輯交給聰明的Quartz去決定”。基本策略是,
- 如果是隻執行一次的排程,使用MISFIRE_INSTRUCTION_FIRE_NOW
- 如果是無限次的排程(repeatCount是無限的),使用MISFIRE_INSTRUCTION_RESCHEDULE_NEXT_WITH_REMAINING_COUNT
- 否則,使用MISFIRE_INSTRUCTION_RESCHEDULE_NOW_WITH_EXISTING_REPEAT_COUNT
MisFire的東西挺繁雜的,可以參考這篇
Calendar
這裡的Calendar不是jdk的java.util.Calendar,不是為了計算日期的。它的作用是在於補充Trigger的時間。可以排除或加入某一些特定的時間點。
以”每月25日零點自動還卡債“為例,我們想排除掉每年的2月25號零點這個時間點(因為有2.14,所以2月一定會破產)。這個時間,就可以用Calendar來實現。
例子:
AnnualCalendar cal = new AnnualCalendar(); //定義一個每年執行Calendar,精度為天,即不能定義到2.25號下午2:00
java.util.Calendar excludeDay = new GregorianCalendar();
excludeDay.setTime(newDate().inMonthOnDay(2, 25).build());
cal.setDayExcluded(excludeDay, true); //設定排除2.25這個日期
scheduler.addCalendar("FebCal", cal, false, false); //scheduler加入這個Calendar
//定義一個Trigger
Trigger trigger = newTrigger().withIdentity("trigger1", "group1")
.startNow()//一旦加入scheduler,立即生效
.modifiedByCalendar("FebCal") //使用Calendar !!
.withSchedule(simpleSchedule()
.withIntervalInSeconds(1)
.repeatForever())
.build();
Quartz體貼地為我們提供以下幾種Calendar,注意,所有的Calendar既可以是排除,也可以是包含,取決於:
- HolidayCalendar。指定特定的日期,比如20140613。精度到天。
- DailyCalendar。指定每天的時間段(rangeStartingTime, rangeEndingTime),格式是HH:MM[:SS[:mmm]]。也就是最大精度可以到毫秒。
- WeeklyCalendar。指定每星期的星期幾,可選值比如為java.util.Calendar.SUNDAY。精度是天。
- MonthlyCalendar。指定每月的幾號。可選值為1-31。精度是天
- AnnualCalendar。 指定每年的哪一天。使用方式如上例。精度是天。
- CronCalendar。指定Cron表示式。精度取決於Cron表示式,也就是最大精度可以到秒。
Trigger實現類
Quartz有以下幾種Trigger實現:
SimpleTrigger
指定從某一個時間開始,以一定的時間間隔(單位是毫秒)執行的任務。
它適合的任務類似於:9:00 開始,每隔1小時,執行一次。
它的屬性有:
- repeatInterval 重複間隔
- repeatCount 重複次數。實際執行次數是 repeatCount+1。因為在startTime的時候一定會執行一次。** 下面有關repeatCount 屬性的都是同理。 **
例子:
simpleSchedule()
.withIntervalInHours(1) //每小時執行一次
.repeatForever() //次數不限
.build();
simpleSchedule()
.withIntervalInMinutes(1) //每分鐘執行一次
.withRepeatCount(10) //次數為10次
.build();
CalendarIntervalTrigger
類似於SimpleTrigger,指定從某一個時間開始,以一定的時間間隔執行的任務。 但是不同的是SimpleTrigger指定的時間間隔為毫秒,沒辦法指定每隔一個月執行一次(每月的時間間隔不是固定值),而CalendarIntervalTrigger支援的間隔單位有秒,分鐘,小時,天,月,年,星期。
相較於SimpleTrigger有兩個優勢:1、更方便,比如每隔1小時執行,你不用自己去計算1小時等於多少毫秒。 2、支援不是固定長度的間隔,比如間隔為月和年。但劣勢是精度只能到秒。
它適合的任務類似於:9:00 開始執行,並且以後每週 9:00 執行一次
它的屬性有:
- interval 執行間隔
- intervalUnit 執行間隔的單位(秒,分鐘,小時,天,月,年,星期)
例子:
calendarIntervalSchedule()
.withIntervalInDays(1) //每天執行一次
.build();
calendarIntervalSchedule()
.withIntervalInWeeks(1) //每週執行一次
.build();
DailyTimeIntervalTrigger
指定每天的某個時間段內,以一定的時間間隔執行任務。並且它可以支援指定星期。
它適合的任務類似於:指定每天9:00 至 18:00 ,每隔70秒執行一次,並且只要週一至週五執行。
它的屬性有:
- startTimeOfDay 每天開始時間
- endTimeOfDay 每天結束時間
- daysOfWeek 需要執行的星期
- interval 執行間隔
- intervalUnit 執行間隔的單位(秒,分鐘,小時,天,月,年,星期)
- repeatCount 重複次數
例子:
dailyTimeIntervalSchedule()
.startingDailyAt(TimeOfDay.hourAndMinuteOfDay(9, 0)) //第天9:00開始
.endingDailyAt(TimeOfDay.hourAndMinuteOfDay(16, 0)) //16:00 結束
.onDaysOfTheWeek(MONDAY,TUESDAY,WEDNESDAY,THURSDAY,FRIDAY) //週一至週五執行
.withIntervalInHours(1) //每間隔1小時執行一次
.withRepeatCount(100) //最多重複100次(實際執行100+1次)
.build();
dailyTimeIntervalSchedule()
.startingDailyAt(TimeOfDay.hourAndMinuteOfDay(9, 0)) //第天9:00開始
.endingDailyAfterCount(10) //每天執行10次,這個方法實際上根據 startTimeOfDay+interval*count 算出 endTimeOfDay
.onDaysOfTheWeek(MONDAY,TUESDAY,WEDNESDAY,THURSDAY,FRIDAY) //週一至週五執行
.withIntervalInHours(1) //每間隔1小時執行一次
.build();
CronTrigger
適合於更復雜的任務,它支援型別於Linux Cron的語法(並且更強大)。基本上它覆蓋了以上三個Trigger的絕大部分能力(但不是全部)—— 當然,也更難理解。
它適合的任務類似於:每天0:00,9:00,18:00各執行一次。
它的屬性只有:
- Cron表示式。但這個表示式本身就夠複雜了。下面會有說明。
例子:
cronSchedule("0 0/2 8-17 * * ?") // 每天8:00-17:00,每隔2分鐘執行一次
.build();
cronSchedule("0 30 9 ? * MON") // 每週一,9:30執行一次
.build();
weeklyOnDayAndHourAndMinute(MONDAY,9, 30) //等同於 0 30 9 ? * MON
.build();
Cron表示式
位置 | 時間域 | 允許值 | 特殊值 |
---|---|---|---|
1 | 秒 | 0-59 | , - * / |
2 | 分鐘 | 0-59 | , - * / |
3 | 小時 | 0-23 | , - * / |
4 | 日期 | 1-31 | , - * ? / L W C |
5 | 月份 | 1-12 | , - * / |
6 | 星期 | 1-7 | , - * ? / L C # |
7 | 年份(可選) | 1-31 | , - * / |
星號(*):可用在所有欄位中,表示對應時間域的每一個時刻,例如, 在分鐘欄位時,表示“每分鐘”;
問號(?):該字元只在日期和星期欄位中使用,它通常指定為“無意義的值”,相當於點位符;
減號(-):表達一個範圍,如在小時欄位中使用“10-12”,則表示從10到12點,即10,11,12;
逗號(,):表達一個列表值,如在星期欄位中使用“MON,WED,FRI”,則表示星期一,星期三和星期五;
斜槓(/):x/y表達一個等步長序列,x為起始值,y為增量步長值。如在分鐘欄位中使用0/15,則表示為0,15,30和45秒,而5/15在分鐘欄位中表示5,20,35,50,你也可以使用*/y,它等同於0/y;
L:該字元只在日期和星期欄位中使用,代表“Last”的意思,但它在兩個欄位中意思不同。L在日期欄位中,表示這個月份的最後一天,如一月的31號,非閏年二月的28號;如果L用在星期中,則表示星期六,等同於7。但是,如果L出現在星期欄位裡,而且在前面有一個數值X,則表示“這個月的最後X天”,例如,6L表示該月的最後星期五;
W:該字元只能出現在日期欄位裡,是對前導日期的修飾,表示離該日期最近的工作日。例如15W表示離該月15號最近的工作日,如果該月15號是星期六,則匹配14號星期五;如果15日是星期日,則匹配16號星期一;如果15號是星期二,那結果就是15號星期二。但必須注意關聯的匹配日期不能夠跨月,如你指定1W,如果1號是星期六,結果匹配的是3號星期一,而非上個月最後的那天。W字串只能指定單一日期,而不能指定日期範圍;
LW組合:在日期欄位可以組合使用LW,它的意思是當月的最後一個工作日;
井號(#):該字元只能在星期欄位中使用,表示當月某個工作日。如6#3表示當月的第三個星期五(6表示星期五,#3表示當前的第三個),而4#5表示當月的第五個星期三,假設當月沒有第五個星期三,忽略不觸發;
C:該字元只在日期和星期欄位中使用,代表“Calendar”的意思。它的意思是計劃所關聯的日期,如果日期沒有被關聯,則相當於日曆中所有日期。例如5C在日期欄位中就相當於日曆5日以後的第一天。1C在星期欄位中相當於星期日後的第一天。
Cron表示式對特殊字元的大小寫不敏感,對代表星期的縮寫英文大小寫也不敏感。
Cron表示式案例
表示式 | 說明 |
---|---|
0 0 12 * * ? | 每天12點執行 |
0 15 10 ? * * | 每天10:15執行 |
0 15 10 * * ? | 每天10:15執行 |
0 15 10 * * ? * | 每天10:15執行 |
0 15 10 * * ? 2008 | 在2008年的每天10:15執行 |
0 * 14 * * ? | 每天14點到15點之間每分鐘執行一次,開始於14:00,結束於14:59。 |
0 0/5 14 * * ? | 每天14點到15點每5分鐘執行一次,開始於14:00,結束於14:55。 |
0 0/5 14,18 * * ? | 每天14點到15點每5分鐘執行一次,此外每天18點到19點每5鍾也執行一次。 |
0 0-5 14 * * ? | 每天14:00點到14:05,每分鐘執行一次。 |
0 10,44 14 ? 3 WED | 3月每週三的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 2007-2009 | 在2007,2008,2009年每個月的最後一個星期五的10:15分執行。 |
0 15 10 ? * 6#3 | 每月第三個星期五的10:15分執行。 |
CronTrigger的熄火策略
- MISFIRE_INSTRUCTION_FIRE_ONCE_NOW - 如果熄火,該指令會告訴Quartz希望馬上再次觸發
- MISFIRE_INSTRUCTION_DO_NOTHING - 如果熄火,該指令會告訴Quartz下一次執行時間到來時再執行,並不想馬上執行
一個cronTrigger的簡單案例
Trigger t1 = TriggerBuilder.newTrigger()
.withIdentity("trigger1", "group1")
.startAt(startTime).endAt(endTime)
.withSchedule(CronScheduleBuilder.cronSchedule("0 0 12 * * ? ")
)
.build();
相關文章
- java 定時任務 quartz 時間表示式Cron總結Javaquartz
- 【Java定時任務】淺談CronTrigger的用法和線上Cron表示式生成網址Java
- Quartz定時任務框架(二) Quartz詳解quartz框架
- GO的定時器Timer 和定時任務cronGo定時器
- Golang——Cron 定時任務Golang
- Quartz定時任務框架(一) 入門案例quartz框架
- 使用 Cron4j 表示式 在 Solon 裡開發定時任務
- 石英定時任務-quartzquartz
- linux定時任務cron配置Linux
- 淺談分散式定時任務之quartz分散式quartz
- 簡單說說Java 定時任務框架---QuartzJava框架quartz
- quartz學習-quartz編碼方式實現定時任務簡例quartz
- crontab和cron表示式詳解
- SpringBoot整合Quartz定時任務Spring Bootquartz
- Quartz 定時任務管理類quartz
- 排查linux 定時任務cron crontabLinux
- DolphinScheduler心臟:Quartz的定時任務排程框架深度解析quartz框架
- java Quartz 定時任務管理類Javaquartz
- Spring Boot 配置 Quartz 定時任務Spring Bootquartz
- springboot Quartz 定時任務工具類Spring Bootquartz
- Laravel Cron 定時任務 “跳坑” 點Laravel
- Quartz - Spring整合Quartz實現叢集的定時任務quartzSpring
- Java中的定時任務最佳化:從Cron表示式到高精度排程的實現Java
- 普通web整合quartz跑定時任務Webquartz
- Quartz高可用定時任務快速上手quartz
- [原始碼分析] 定時任務排程框架 Quartz 之 故障切換原始碼框架quartz
- 3分鐘掌握Quartz.net分散式定時任務的姿勢quartz分散式
- 震驚!Windows Service服務和定時任務框架quartz之間原來是這種關係……Windows框架quartz
- Cron 表示式
- cron表示式
- Spring 定時任務Scheduled 開發詳細圖文Spring
- 【親測有效】【定時】定時任務 @Scheduled(cron = "0 0 21 * * ?") 【Scheduled失效】
- springboot整合Quartz實現動態配置定時任務Spring Bootquartz
- Grails整合Quartz外掛實現定時任務(Job)AIquartz
- 專案實戰!接入分散式定時任務框架分散式框架
- springboot+quartz以持久化的方式實現定時任務Spring Bootquartz持久化
- laravel框架任務排程(定時執行任務)Laravel框架
- 在 Linux 中安全且輕鬆地管理 Cron 定時任務Linux