Spring @Scheduler使用cron表達(dá)式時(shí)的執(zhí)行問(wèn)題詳解
前言
Spring Scheduler里有兩個(gè)概念:任務(wù)(Task)和運(yùn)行任務(wù)的框架(TaskExecutor/TaskScheduler)。TaskExecutor顧名思義,是任務(wù)的執(zhí)行器,允許我們異步執(zhí)行多個(gè)任務(wù)。TaskScheduler是任務(wù)調(diào)度器,來(lái)運(yùn)行未來(lái)的定時(shí)任務(wù)。觸發(fā)器Trigger可以決定定時(shí)任務(wù)是否該運(yùn)行了,最常用的觸發(fā)器是CronTrigger。Spring內(nèi)置了多種類型的TaskExecutor和TaskScheduler,方便用戶根據(jù)不同業(yè)務(wù)場(chǎng)景選擇。
本文主要介紹了關(guān)于Spring @Scheduler使用cron表達(dá)式執(zhí)行問(wèn)題的相關(guān)內(nèi)容,分享出來(lái)供大家參考學(xué)習(xí),下面話不多說(shuō)了,來(lái)一起看看詳細(xì)的介紹吧
主要想弄清使用Spring @Scheduler cron表達(dá)式時(shí)的兩個(gè)問(wèn)題:
- 同一定時(shí)任務(wù),第二次觸發(fā)時(shí)間到了,第一次還沒(méi)有執(zhí)行完成時(shí)會(huì)執(zhí)行嗎?
- 不同的定時(shí)任務(wù),相互之間是否有影響?
結(jié)論寫在前面:
- 同一定時(shí)任務(wù),第二次觸發(fā)時(shí)間到了,第一次還沒(méi)有執(zhí)行完成時(shí)會(huì)執(zhí)行嗎?不會(huì),會(huì)等前一次執(zhí)行完成才執(zhí)行下一次
- 不同的定時(shí)任務(wù),相互之間是否有影響?取決于可用的定時(shí)任務(wù)線程數(shù),如果線程數(shù)足夠則不會(huì)影響;如果可用定時(shí)任務(wù)線程數(shù)少于要執(zhí)行定時(shí)任務(wù)數(shù)量,未能獲取到線程的自然要等到有空閑線程時(shí)才能執(zhí)行。
下面是實(shí)驗(yàn)過(guò)程。。。。。
使用Spring @Scheduler 時(shí),默認(rèn)只有一個(gè)線程,針對(duì)上面的問(wèn)題,設(shè)計(jì)了3個(gè)實(shí)驗(yàn):
- 設(shè)置Scheduler為多線程,設(shè)置一個(gè)線程5秒執(zhí)行一次,方法體為 sleep8秒
- 使用Scheduler默認(rèn)的單線程,設(shè)置兩個(gè)線程都是5秒執(zhí)行一次,一個(gè) sleep8秒,一個(gè)不sleep
- 設(shè)置Scheduler為多線程,設(shè)置兩個(gè)線程都是5秒執(zhí)行一次,一個(gè) sleep8秒,一個(gè)不sleep
實(shí)驗(yàn)一
設(shè)置Scheduler為多線程,設(shè)置一個(gè)線程5秒執(zhí)行一次,方法體為 sleep8秒:
@Scheduled(cron = "*/5 * * * * *")
public void test1() throws InterruptedException {
log.info("test1, 5秒執(zhí)行一次,每次執(zhí)行sleep 8s");
Thread.sleep(8000L);
}
結(jié)果:
2017-10-11 17:49:45 scheduler-1 test1, 5秒執(zhí)行一次,每次執(zhí)行sleep 8
2017-10-11 17:49:55 scheduler-1 test1, 5秒執(zhí)行一次,每次執(zhí)行sleep 8
2017-10-11 17:50:05 scheduler-1 test1, 5秒執(zhí)行一次,每次執(zhí)行sleep 8
2017-10-11 17:50:15 scheduler-2 test1, 5秒執(zhí)行一次,每次執(zhí)行sleep 8
2017-10-11 17:50:25 scheduler-2 test1, 5秒執(zhí)行一次,每次執(zhí)行sleep 8
2017-10-11 17:50:35 scheduler-1 test1, 5秒執(zhí)行一次,每次執(zhí)行sleep 8
結(jié)論:
@Scheduled使用cron表達(dá)式,設(shè)置為多線程時(shí),同一任務(wù)前一次沒(méi)有執(zhí)行完成,不會(huì)執(zhí)行下一次
實(shí)驗(yàn)二
使用Scheduler默認(rèn)的單線程,設(shè)置兩個(gè)線程都是5秒執(zhí)行一次,一個(gè) sleep8秒,一個(gè)不sleep
如果test2每8秒執(zhí)行一次,則為串行
@Scheduled(cron = "*/5 * * * * *")
public void test1() throws InterruptedException {
System.out.println("test1, 5秒執(zhí)行一次,每次執(zhí)行sleep 8s");
Thread.sleep(8000L);
}
@Scheduled(cron = "*/5 * * * * *")
public void test2() {
System.out.println("test2, 5秒執(zhí)行一次,不sleep");
}
執(zhí)行結(jié)果:
2017-10-11 17:17:35 test2, 5秒執(zhí)行一次,不sleep
2017-10-11 17:17:35 test1, 5秒執(zhí)行一次,每次執(zhí)行sleep 8s
2017-10-11 17:17:43 test2, 5秒執(zhí)行一次,不sleep
2017-10-11 17:17:45 test1, 5秒執(zhí)行一次,每次執(zhí)行sleep 8s
2017-10-11 17:17:53 test2, 5秒執(zhí)行一次,不sleep
2017-10-11 17:17:55 test2, 5秒執(zhí)行一次,不sleep
2017-10-11 17:17:55 test1, 5秒執(zhí)行一次,每次執(zhí)行sleep 8s
2017-10-11 17:18:03 test2, 5秒執(zhí)行一次,不sleep
2017-10-11 17:18:05 test2, 5秒執(zhí)行一次,不sleep
2017-10-11 17:18:05 test1, 5秒執(zhí)行一次,每次執(zhí)行sleep 8s
2017-10-11 17:18:13 test2, 5秒執(zhí)行一次,不sleep
2017-10-11 17:18:15 test1, 5秒執(zhí)行一次,每次執(zhí)行sleep 8s
2017-10-11 17:18:23 test2, 5秒執(zhí)行一次,不sleep
2017-10-11 17:18:25 test1, 5秒執(zhí)行一次,每次執(zhí)行sleep 8s
對(duì)比期望執(zhí)行時(shí)間:
| 執(zhí)行次數(shù) | task | 期望執(zhí)行時(shí)間 | 實(shí)際執(zhí)行時(shí)間 |
|---|---|---|---|
| 1 | task1 | 17:17:35 | 17:17:35 |
| 1 | task2 | 17:17:35 | 17:17:35 |
| 2 | task1 | 17:17:40 | 17:17:43 |
| 2 | task2 | 17:17:40 | 17:17:45 |
結(jié)論:
@Scheduled使用cron表達(dá)式 ,配置為一個(gè)線程時(shí),不同定時(shí)任務(wù)是串行執(zhí)行,且上次沒(méi)有執(zhí)行完時(shí)不會(huì)執(zhí)行下次
實(shí)驗(yàn)三
設(shè)置Scheduler為多線程,設(shè)置兩個(gè)線程都是5秒執(zhí)行一次,一個(gè) sleep8秒,一個(gè)不sleep
@Scheduled(cron = "*/5 * * * * *")
public void test1() throws InterruptedException {
log.info("test1, 5秒執(zhí)行一次,每次執(zhí)行sleep 8s");
Thread.sleep(8000L);
}
@Scheduled(cron = "*/5 * * * * *")
public void test2() {
log.info("test2, 5秒執(zhí)行一次,不sleep");
}
結(jié)果:
2017-10-11 18:12:40 scheduler-2 test2, 5秒執(zhí)行一次,不sleep
2017-10-11 18:12:40 scheduler-1 test1, 5秒執(zhí)行一次,每次執(zhí)行sleep 8s
2017-10-11 18:12:45 scheduler-2 test2, 5秒執(zhí)行一次,不sleep
2017-10-11 18:12:50 scheduler-1 test1, 5秒執(zhí)行一次,每次執(zhí)行sleep 8s
2017-10-11 18:12:50 scheduler-2 test2, 5秒執(zhí)行一次,不sleep
2017-10-11 18:12:55 scheduler-2 test2, 5秒執(zhí)行一次,不sleep
2017-10-11 18:13:00 scheduler-1 test1, 5秒執(zhí)行一次,每次執(zhí)行sleep 8s
對(duì)比期望執(zhí)行時(shí)間:
| 執(zhí)行次數(shù) | task | 期望執(zhí)行時(shí)間 | 實(shí)際執(zhí)行時(shí)間 |
|---|---|---|---|
| 1 | task1 | 18:12:40 | 18:12:40 |
| 1 | task2 | 18:12:40 | 18:12:40 |
| 2 | task1 | 18:12:45 | 18:12:50 |
| 2 | task2 | 18:12:45 | 18:12:45 |
結(jié)論:
@Scheduled使用cron表達(dá)式 ,配置為多線程時(shí),不同定時(shí)任務(wù)不是串行執(zhí)行,且上次沒(méi)有執(zhí)行完時(shí)不會(huì)執(zhí)行下次
設(shè)置定時(shí)任務(wù)為多線程
這里用的是spring boot:
@Configuration
public class ScheduleConfig {
@Bean
public ThreadPoolTaskScheduler threadPoolTaskScheduler() {
ThreadPoolTaskScheduler scheduler = new ThreadPoolTaskScheduler();
scheduler.setPoolSize(3);
scheduler.setThreadNamePrefix("scheduler-");
return scheduler;
}
}
總結(jié)
以上就是這篇文章的全部?jī)?nèi)容了,希望本文的內(nèi)容對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,如果有疑問(wèn)大家可以留言交流,謝謝大家對(duì)腳本之家的支持。
相關(guān)文章
Java服務(wù)端如何解決跨域問(wèn)題?CORS請(qǐng)求頭方式
這篇文章主要介紹了Java服務(wù)端如何解決跨域問(wèn)題?CORS請(qǐng)求頭方式,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2022-11-11
SpringBoot常見(jiàn)錯(cuò)誤圖文總結(jié)
最近在使用idea+Springboot開發(fā)項(xiàng)目中遇到一些問(wèn)題,這篇文章主要給大家介紹了關(guān)于SpringBoot常見(jiàn)錯(cuò)誤總結(jié)的相關(guān)資料,文中通過(guò)實(shí)例代碼介紹的非常詳細(xì),需要的朋友可以參考下2023-06-06
基于Springboot+Vue實(shí)現(xiàn)的在線答題闖關(guān)系統(tǒng)全過(guò)程
這篇文章主要介紹了基于Springboot+Vue實(shí)現(xiàn)的在線答題闖關(guān)系統(tǒng)的相關(guān)資料,文中包括前端Vue.js、后端SpringBoot及MySQL數(shù)據(jù)庫(kù)的使用,系統(tǒng)功能涵蓋順序出題、體型練習(xí)、隨機(jī)出題、錯(cuò)題本、收藏題和答題統(tǒng)計(jì)等,需要的朋友可以參考下2024-12-12
spring?cloud?eureka注冊(cè)原理-注冊(cè)失敗填坑筆記
這篇文章主要介紹了spring?cloud?eureka注冊(cè)原理-注冊(cè)失敗填坑筆記,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2022-05-05
SpringBoot整合Redis實(shí)現(xiàn)緩存分頁(yè)數(shù)據(jù)查詢功能
類似淘寶首頁(yè),這些商品是從數(shù)據(jù)庫(kù)中查出來(lái)的嗎,答案肯定不是,本文我們就通過(guò)一個(gè)案例實(shí)操一下,首頁(yè)熱點(diǎn)數(shù)據(jù)怎么放到Redis中去查詢,感興趣的同學(xué)可以參考一下2023-06-06
Spring整合Quartz分布式調(diào)度的示例代碼
本篇文章主要介紹了Spring整合Quartz分布式調(diào)度的示例代碼,小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2018-04-04

