眨眼的功夫,就把定時(shí)任務(wù)的三種調(diào)度策略說(shuō)得明明白白
Spring Task 無(wú)疑是 Spring 環(huán)境下單機(jī)定時(shí)任務(wù)的首選。它用起來(lái)非常簡(jiǎn)單,功能也夠用。
Spring Task 有三種模式,分別是:fixedDelay、cron 和 fixedRate。話(huà)不多說(shuō),我們先看代碼:
public class TimeTask {
private int[] people = {6, 2, 3, 1};
private int count = 0;
(fixedDelay = 5000)
public void fixedDelayTask() throws InterruptedException {
if (count < 4) {
int timeConsuming = people[count];
log.info("fixedDelayTask-----第 {} 個(gè)人在 {} 開(kāi)始如廁,耗時(shí):{} 秒", count + 1,
formatTime(),
timeConsuming);
Thread.sleep(timeConsuming * 1000L);
count++;
}
}
(cron = "0/5 * * * * ? ")
public void cronTask() throws InterruptedException {
if (count < 4) {
int timeConsuming = people[count];
log.info("cronTask-----第 {} 個(gè)人在 {} 開(kāi)始如廁,耗時(shí):{} 秒", count + 1,
formatTime(),
timeConsuming);
Thread.sleep(timeConsuming * 1000L);
count++;
}
}
(fixedRate = 5000)
public void fixedRateTask() throws InterruptedException {
if (count < 4) {
int timeConsuming = people[count];
log.info("fixedRateTask-----第 {} 個(gè)人在 {} 開(kāi)始如廁,耗時(shí):{} 秒", count + 1,
formatTime(),
timeConsuming);
Thread.sleep(timeConsuming * 1000L);
count++;
}
}
private String formatTime() {
return LocalTime.now().format(DateTimeFormatter.ofPattern("HH:mm:ss"));
}
}
這三種模式的用法都很簡(jiǎn)單,使用方式也很類(lèi)似。那么它們究竟有什么不同呢?下面我們通過(guò)一個(gè)五星級(jí)豪華公測(cè)的故事來(lái)說(shuō)明一下。
故事背景
話(huà)說(shuō)某地有一個(gè)五星級(jí)豪華廁所,大家都喜歡來(lái)這里如廁。因此坑位經(jīng)常供不應(yīng)求,需要排隊(duì)如廁。一天,廁所外有四個(gè)人排隊(duì),每個(gè)人如廁需要的時(shí)間如下:
- 第一個(gè)人 6 秒鐘
- 第二個(gè)人 2 秒鐘
- 第三個(gè)人 3 秒鐘
- 第四個(gè)人 1 秒鐘
從第一個(gè)人開(kāi)始如廁進(jìn)行計(jì)時(shí)。
fixedDelay 模式
日志輸出:
fixedDelayTask-----第 1 個(gè)人在 18:07:23 開(kāi)始如廁,耗時(shí):6 秒
fixedDelayTask-----第 2 個(gè)人在 18:07:34 開(kāi)始如廁,耗時(shí):2 秒
fixedDelayTask-----第 3 個(gè)人在 18:07:41 開(kāi)始如廁,耗時(shí):3 秒
fixedDelayTask-----第 4 個(gè)人在 18:07:49 開(kāi)始如廁,耗時(shí):1 秒
@Scheduled(fixedDelay = 5000)。
廁所在該模式下有一個(gè)特點(diǎn):每次用完廁所后,需要有 5 秒鐘的廁所自潔時(shí)間,需要對(duì)廁所進(jìn)行清潔消毒等工作,從而保證下次使用的時(shí)候依然干凈衛(wèi)生。執(zhí)行情況如圖所示:
- 第一個(gè)人在第 0 秒時(shí)開(kāi)始如廁,6 秒后結(jié)束,廁所需要 5 秒鐘的自潔時(shí)間。
- 第二個(gè)人在第 11 秒(6+5)時(shí)開(kāi)始如廁,2 秒后結(jié)束,廁所需要 5 秒鐘的自潔時(shí)間。
- 第三個(gè)人在第 18 秒(11+2+5)時(shí)開(kāi)始如廁,3 秒后結(jié)束,廁所需要 5 秒鐘的自潔時(shí)間。
- 第四個(gè)人在第 26 秒 (18+3+5)時(shí)開(kāi)始如廁,1 秒后結(jié)束…
Cron 模式
日志輸出:
cronTask-----第 1 個(gè)人在 18:09:15 開(kāi)始如廁,耗時(shí):6 秒
cronTask-----第 2 個(gè)人在 18:09:25 開(kāi)始如廁,耗時(shí):2 秒
cronTask-----第 3 個(gè)人在 18:09:30 開(kāi)始如廁,耗時(shí):3 秒
cronTask-----第 4 個(gè)人在 18:09:35 開(kāi)始如廁,耗時(shí):1 秒
@Scheduled(cron = "0/5 * * * * ? ")。
在該模式下,廁所只在時(shí)間秒數(shù) 5 的整數(shù)倍時(shí)準(zhǔn)許人員進(jìn)入使用。因?yàn)橥ㄟ^(guò)嚴(yán)謹(jǐn)?shù)目茖W(xué)分析,發(fā)現(xiàn)在秒數(shù)為 5 的整數(shù)倍時(shí)如廁體驗(yàn)更佳,所以只有當(dāng)前時(shí)間秒數(shù)為 5 的整數(shù)倍時(shí)才可以進(jìn)入。并且,五星級(jí)豪華公廁升級(jí)設(shè)備,可以在如廁完成的瞬間完成自潔消毒,因此不再需要額外的自潔時(shí)間了,也提升了廁所利用率。執(zhí)行情況如圖所示:
- 第一個(gè)人在 18:09:15 時(shí)開(kāi)始如廁,6 秒后(18:09:21)結(jié)束,下一個(gè)如廁吉時(shí)為 18:09:25。
- 第二個(gè)人在 18:09:25 時(shí)開(kāi)始如廁,2 秒后(18:09:27)結(jié)束,下一個(gè)如廁吉時(shí)為 18:09:30。
- 第三個(gè)人在 18:09:30 時(shí)開(kāi)始如廁,3 秒后(18:09:33)結(jié)束,下一個(gè)如廁吉時(shí)為 18:09:35。
- 第四個(gè)人在 18:09:35 時(shí)開(kāi)始如廁,1 秒后(18:09:36)結(jié)束…
fixedRate 模式
日志輸出:
fixedRateTask-----第 1 個(gè)人在 18:10:18 開(kāi)始如廁,耗時(shí):6 秒
fixedRateTask-----第 2 個(gè)人在 18:10:24 開(kāi)始如廁,耗時(shí):2 秒
fixedRateTask-----第 3 個(gè)人在 18:10:28 開(kāi)始如廁,耗時(shí):3 秒
fixedRateTask-----第 4 個(gè)人在 18:10:33 開(kāi)始如廁,耗時(shí):1 秒
@Scheduled(fixedRate = 5000)。
經(jīng)過(guò)長(zhǎng)時(shí)間的大數(shù)據(jù)分析,得出一個(gè)結(jié)論——人的最佳如廁時(shí)長(zhǎng)是 5 秒鐘。所以在該模式下,人們?nèi)鐜?,廁所?huì)根據(jù)等待人數(shù)提前制定出如廁計(jì)劃,即為每位等待者分配 5 秒鐘如廁時(shí)間。但是有一個(gè)規(guī)則:當(dāng)如廁者提前結(jié)束,那么下一個(gè)人仍然需要等夠 5 秒鐘;而當(dāng)如廁者超時(shí)以后,待廁者可以在上一人完成時(shí)立即如廁。那么:
如廁計(jì)劃如下:
- 第一個(gè)人:第 0 秒進(jìn)入
- 第二個(gè)人:第 5 秒進(jìn)入
- 第三個(gè)人:第 10 秒進(jìn)入
- 第四個(gè)人:第 15 秒進(jìn)入
根據(jù)故事背景中每個(gè)人如廁時(shí)間,實(shí)際情況如圖所示:
- 第一個(gè)人在第 0 秒時(shí),即 18:10:18 時(shí)開(kāi)始如廁,6 秒后結(jié)束,超時(shí),第二個(gè)人無(wú)縫如廁;
- 第二個(gè)人在第 6 秒(0+6)時(shí),即 18:10:24 開(kāi)始如廁,2 秒后結(jié)束,未超時(shí),第三個(gè)人等待 2 秒,按計(jì)劃時(shí)間如廁;
- 第三個(gè)人在第 10 秒(6+2+2)時(shí),即 18:10:28 開(kāi)始如廁,3 秒后結(jié)束,未超時(shí),第四個(gè)人等待 2 秒按計(jì)劃如廁;
- 第四個(gè)人在第 15 秒 (6+2+2+3+2)時(shí),即 18:10:33 開(kāi)始如廁,1 秒后結(jié)束…
Cron 表達(dá)式
Cron 模式是定時(shí)任務(wù)中最強(qiáng)大的觸發(fā)策略,可以應(yīng)對(duì)更多的情況。一個(gè) Cron 表達(dá)式總共有 7 個(gè)元素,分別如下表所示:
時(shí)間單位 | 是否必填 | 取值范圍 | 通配符 |
秒 | 是 | 0-59 的整數(shù) | , - * / 四個(gè)字符 |
分 | 是 | 0-59 的整數(shù) | , - * / 四個(gè)字符 |
時(shí) | 是 | 0-23 的整數(shù) | , - * / 四個(gè)字符 |
日 | 是 | 1-31 的整數(shù)(需要考慮該月的具體天數(shù)) | ,- * ? / L W C 八個(gè)字符 |
月 | 是 | 1~12 的整數(shù)或者 JAN-DEC | , - * / 四個(gè)字符 |
周 | 是 | 1~7 的整數(shù)或者 SUN-SAT (1=SUN) | , - * ? / L C # 八個(gè)字符 |
年 | 否 | 1970~2099 | , - * / 四個(gè)字符 |