Spring Boot 2.x基礎(chǔ)教程之使用@Scheduled實(shí)現(xiàn)定時(shí)任務(wù)的方法
我們?cè)诰帉?xiě)Spring Boot應(yīng)用中經(jīng)常會(huì)遇到這樣的場(chǎng)景,比如:我需要定時(shí)地發(fā)送一些短信、郵件之類(lèi)的操作,也可能會(huì)定時(shí)地檢查和監(jiān)控一些標(biāo)志、參數(shù)等。
創(chuàng)建定時(shí)任務(wù)
在Spring Boot中編寫(xiě)定時(shí)任務(wù)是非常簡(jiǎn)單的事,下面通過(guò)實(shí)例介紹如何在Spring Boot中創(chuàng)建定時(shí)任務(wù),實(shí)現(xiàn)每過(guò)5秒輸出一下當(dāng)前時(shí)間。
在Spring Boot的主類(lèi)中加入@EnableScheduling注解,啟用定時(shí)任務(wù)的配置
@SpringBootApplication
@EnableScheduling
public class Application {
public static void main(String[] args) {
SpringApplication.run(Application.class, args);
}
}
創(chuàng)建定時(shí)任務(wù)實(shí)現(xiàn)類(lèi)
@Component
public class ScheduledTasks {
private static final SimpleDateFormat dateFormat = new SimpleDateFormat("HH:mm:ss");
@Scheduled(fixedRate = 5000)
public void reportCurrentTime() {
log.info("現(xiàn)在時(shí)間:" + dateFormat.format(new Date()));
}
}
運(yùn)行程序,控制臺(tái)中可以看到類(lèi)似如下輸出,定時(shí)任務(wù)開(kāi)始正常運(yùn)作了。
2021-07-13 14:56:56.413 INFO 34836 --- [ main] c.d.chapter71.Chapter71Application : Started Chapter71Application in 1.457 seconds (JVM running for 1.835)
2021-07-13 14:57:01.411 INFO 34836 --- [ scheduling-1] com.didispace.chapter71.ScheduledTasks : 現(xiàn)在時(shí)間:14:57:01
2021-07-13 14:57:06.412 INFO 34836 --- [ scheduling-1] com.didispace.chapter71.ScheduledTasks : 現(xiàn)在時(shí)間:14:57:06
2021-07-13 14:57:11.413 INFO 34836 --- [ scheduling-1] com.didispace.chapter71.ScheduledTasks : 現(xiàn)在時(shí)間:14:57:11
2021-07-13 14:57:16.413 INFO 34836 --- [ scheduling-1] com.didispace.chapter71.ScheduledTasks : 現(xiàn)在時(shí)間:14:57:16
@Scheduled詳解
在上面的入門(mén)例子中,使用了@Scheduled(fixedRate = 5000) 注解來(lái)定義每過(guò)5秒執(zhí)行的任務(wù)。對(duì)于@Scheduled的使用,我們從源碼里看看有哪些配置:
@Target({ElementType.METHOD, ElementType.ANNOTATION_TYPE})
@Retention(RetentionPolicy.RUNTIME)
@Documented
@Repeatable(Schedules.class)
public @interface Scheduled {
String CRON_DISABLED = ScheduledTaskRegistrar.CRON_DISABLED;
String cron() default "";
String zone() default "";
long fixedDelay() default -1;
String fixedDelayString() default "";
long fixedRate() default -1;
String fixedRateString() default "";
long initialDelay() default -1;
String initialDelayString() default "";
}
這些具體配置信息的含義如下:
- cron:通過(guò)cron表達(dá)式來(lái)配置執(zhí)行規(guī)則
- zone:cron表達(dá)式解析時(shí)使用的時(shí)區(qū)
- fixedDelay:上一次執(zhí)行結(jié)束到下一次執(zhí)行開(kāi)始的間隔時(shí)間(單位:ms)
- fixedDelayString:上一次任務(wù)執(zhí)行結(jié)束到下一次執(zhí)行開(kāi)始的間隔時(shí)間,使用java.time.Duration#parse解析
- fixedRate:以固定間隔執(zhí)行任務(wù),即上一次任務(wù)執(zhí)行開(kāi)始到下一次執(zhí)行開(kāi)始的間隔時(shí)間(單位:ms),若在調(diào)度任務(wù)執(zhí)行時(shí),上一次任務(wù)還未執(zhí)行完畢,會(huì)加入worker隊(duì)列,等待上一次執(zhí)行完成后立即執(zhí)行下一次任務(wù)
- fixedRateString:與fixedRate邏輯一致,只是使用java.time.Duration#parse解析
- initialDelay:首次任務(wù)執(zhí)行的延遲時(shí)間
- initialDelayString:首次任務(wù)執(zhí)行的延遲時(shí)間,使用java.time.Duration#parse解析
思考與進(jìn)階
是不是這樣實(shí)現(xiàn)定時(shí)任務(wù)很簡(jiǎn)單呢?那么繼續(xù)思考一下這種實(shí)現(xiàn)方式是否存在什么弊端呢?
可能初學(xué)者不太容易發(fā)現(xiàn)問(wèn)題,但如果你已經(jīng)有一定的線上項(xiàng)目經(jīng)驗(yàn)的話,問(wèn)題也是顯而易見(jiàn)的:這種模式實(shí)現(xiàn)的定時(shí)任務(wù)缺少在集群環(huán)境下的協(xié)調(diào)機(jī)制。
什么意思呢?假設(shè),我們要實(shí)現(xiàn)一個(gè)定時(shí)任務(wù),用來(lái)每天網(wǎng)上統(tǒng)計(jì)某個(gè)數(shù)據(jù)然后累加到原始數(shù)據(jù)上。我們開(kāi)發(fā)測(cè)試的時(shí)候不會(huì)有問(wèn)題,因?yàn)槎际菃芜M(jìn)程在運(yùn)行的。但是,當(dāng)我們把這樣的定時(shí)任務(wù)部署到生產(chǎn)環(huán)境時(shí),為了更高的可用性,啟動(dòng)多個(gè)實(shí)例是必須的。此時(shí),時(shí)間一到,所有啟動(dòng)的實(shí)例就會(huì)同時(shí)開(kāi)始執(zhí)行這個(gè)任務(wù)。那么問(wèn)題也就出現(xiàn)了,因?yàn)橛欣奂硬僮?,最終我們的結(jié)果就會(huì)出現(xiàn)問(wèn)題。
解決這樣問(wèn)題的方式很多種,比較通用的就是采用分布式鎖的方式,讓同類(lèi)任務(wù)之前的時(shí)候以分布式鎖的方式來(lái)控制執(zhí)行順序,比如:使用Redis、Zookeeper等具備分布式鎖功能的中間件配合就能很好的幫助我們來(lái)協(xié)調(diào)這類(lèi)任務(wù)在集群模式下的執(zhí)行規(guī)則。
代碼示例
本文的完整工程可以查看下面?zhèn)}庫(kù)中的chapter7-1目錄:
Github:https://github.com/dyc87112/SpringBoot-Learning/
Gitee:https://gitee.com/didispace/SpringBoot-Learning/
到此這篇關(guān)于Spring Boot 2.x基礎(chǔ)教程之使用@Scheduled實(shí)現(xiàn)定時(shí)任務(wù)的方法的文章就介紹到這了,更多相關(guān)Spring Boot 2.x定時(shí)任務(wù)內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
- SpringBoot執(zhí)行定時(shí)任務(wù)@Scheduled的方法
- SpringBoot中使用@Scheduled注解創(chuàng)建定時(shí)任務(wù)的實(shí)現(xiàn)
- Spring Boot @Scheduled定時(shí)任務(wù)代碼實(shí)例解析
- Spring boot如何通過(guò)@Scheduled實(shí)現(xiàn)定時(shí)任務(wù)及多線程配置
- springboot 定時(shí)任務(wù)@Scheduled實(shí)現(xiàn)解析
- spring-boot通過(guò)@Scheduled配置定時(shí)任務(wù)及定時(shí)任務(wù)@Scheduled注解的方法
- 詳解Spring Boot中使用@Scheduled創(chuàng)建定時(shí)任務(wù)
相關(guān)文章
Mybatis-plus如何查詢返回對(duì)象內(nèi)有List<String>屬性
在使用Mybatis-Plus進(jìn)行開(kāi)發(fā)時(shí),我們經(jīng)常會(huì)遇到需要處理一對(duì)多關(guān)系映射的情況,例如,查詢用戶數(shù)據(jù)時(shí),可能需要同時(shí)獲取該用戶管理的所有小區(qū)名稱(chēng)列表,這要求我們?cè)诜祷氐膶?shí)體類(lèi)中包含一個(gè)List<String>屬性,用于存放小區(qū)名稱(chēng),實(shí)現(xiàn)這一功能2024-10-10
SpringBoot全局異常處理之解決404/500錯(cuò)誤
在搭建項(xiàng)目框架的時(shí)候用的是springboot,想統(tǒng)一處理異常,但是發(fā)現(xiàn)404的錯(cuò)誤總是捕捉不到,總是返回的是springBoot自帶的錯(cuò)誤結(jié)果信息,這篇文章主要給大家介紹了關(guān)于SpringBoot全局異常處理之解決404/500錯(cuò)誤的相關(guān)資料,需要的朋友可以參考下2023-11-11
springboot集成Swagger的方法(讓你擁有屬于自己的api管理器)
在大型的項(xiàng)目中,如果你有非常多的接口需要統(tǒng)一管理,或者需要進(jìn)行接口測(cè)試,那么我們通常會(huì)在繁雜地api中找到需要進(jìn)行測(cè)試或者管理的接口,接下來(lái)通過(guò)本文給大家介紹springboot集成Swagger的方法讓你擁有屬于自己的api管理器,感興趣的朋友一起看看吧2021-11-11
SpringBoot 簽到獎(jiǎng)勵(lì)實(shí)現(xiàn)方案的示例代碼
這篇文章主要介紹了SpringBoot 簽到獎(jiǎng)勵(lì)實(shí)現(xiàn)方案的示例代碼,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2020-08-08
Java如何使用httpclient檢測(cè)url狀態(tài)及鏈接是否能打開(kāi)
這篇文章主要介紹了Java如何使用httpclient檢測(cè)url狀態(tài)及鏈接是否能打開(kāi),具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2021-09-09
MyBatis-Plus3.x版本使用入門(mén)和踩過(guò)的坑
Mybatis-Plus是Mybatis的增強(qiáng)版,他只是在Mybatis的基礎(chǔ)上增加了功能,且并未對(duì)原有功能進(jìn)行任何的改動(dòng),本文給大家說(shuō)一下MyBatis-Plus3.x版本使用入門(mén)和踩過(guò)的坑,感興趣的朋友跟隨小編一起看看吧2023-10-10
Java操作數(shù)據(jù)庫(kù)(行級(jí)鎖,for update)
這篇文章主要介紹了Java操作數(shù)據(jù)庫(kù)(行級(jí)鎖,for update),文章圍繞Java操作數(shù)據(jù)庫(kù)的相關(guān)資料展開(kāi)詳細(xì)內(nèi)容,需要的小伙伴可以參考一下,希望對(duì)你有所幫助2021-12-12

