詳解SpringBoot定時(shí)任務(wù)說明
1. 定時(shí)任務(wù)實(shí)現(xiàn)方式
定時(shí)任務(wù)實(shí)現(xiàn)方式:
- Java自帶的java.util.Timer類,這個(gè)類允許你調(diào)度一個(gè)java.util.TimerTask任務(wù)。使用這種方式可以讓你的程序按照某一個(gè)頻度執(zhí)行,但不能在指定時(shí)間運(yùn)行。一般用的較少,這篇文章將不做詳細(xì)介紹。
- 使用Quartz,這是一個(gè)功能比較強(qiáng)大的的調(diào)度器,可以讓你的程序在指定時(shí)間執(zhí)行,也可以按照某一個(gè)頻度執(zhí)行,配置起來稍顯復(fù)雜,有空介紹。
- SpringBoot自帶的Scheduled,可以將它看成一個(gè)輕量級(jí)的Quartz,而且使用起來比Quartz簡(jiǎn)單許多,本文主要介紹。
定時(shí)任務(wù)執(zhí)行方式:
- 單線程(串行)
- 多線程(并行)
2. 創(chuàng)建定時(shí)任務(wù)
package com.autonavi.task.test; import org.slf4j.Logger; import org.slf4j.LoggerFactory; import org.springframework.scheduling.annotation.Scheduled; import org.springframework.stereotype.Component; import com.autonavi.task.ScheduledTasks; @Component public class ScheduledTest { private static final Logger logger = LoggerFactory.getLogger(ScheduledTasks.class); @Scheduled(cron="0 0/2 8-20 * * ?") public void executeFileDownLoadTask() { // 間隔2分鐘,執(zhí)行工單上傳任務(wù) Thread current = Thread.currentThread(); System.out.println("定時(shí)任務(wù)1:"+current.getId()); logger.info("ScheduledTest.executeFileDownLoadTask 定時(shí)任務(wù)1:"+current.getId()+ ",name:"+current.getName()); } @Scheduled(cron="0 0/1 8-20 * * ?") public void executeUploadTask() { // 間隔1分鐘,執(zhí)行工單上傳任務(wù) Thread current = Thread.currentThread(); System.out.println("定時(shí)任務(wù)2:"+current.getId()); logger.info("ScheduledTest.executeUploadTask 定時(shí)任務(wù)2:"+current.getId() + ",name:"+current.getName()); } @Scheduled(cron="0 0/3 5-23 * * ?") public void executeUploadBackTask() { // 間隔3分鐘,執(zhí)行工單上傳任務(wù) Thread current = Thread.currentThread(); System.out.println("定時(shí)任務(wù)3:"+current.getId()); logger.info("ScheduledTest.executeUploadBackTask 定時(shí)任務(wù)3:"+current.getId()+ ",name:"+current.getName()); } }
@Scheduled 注解用于標(biāo)注這個(gè)方法是一個(gè)定時(shí)任務(wù)的方法,使用@Scheduled(cron=”…”) 表達(dá)式來設(shè)置定時(shí)任務(wù)。
// 每天早八點(diǎn)到晚八點(diǎn),間隔2分鐘執(zhí)行任務(wù) @Scheduled(cron="0 0/2 8-20 * * ?") // 每天早八點(diǎn)到晚八點(diǎn),間隔3分鐘執(zhí)行任務(wù) @Scheduled(cron="0 0/3 8-20 * * ?") // 每天早八點(diǎn)到晚八點(diǎn),間隔1分鐘執(zhí)行任務(wù) @Scheduled(cron="0 0/1 8-20 * * ?")
3. 啟動(dòng)定時(shí)任務(wù)
@ComponentScan @EnableAutoConfiguration @EnableScheduling @Configuration public class App { private static final Logger logger = LoggerFactory.getLogger(App.class); public static void main(String[] args) { SpringApplication.run(App.class, args); logger.info("oops"); } }
其中 @EnableScheduling 注解的作用是發(fā)現(xiàn)注解@Scheduled的任務(wù)并后臺(tái)執(zhí)行。
4. 執(zhí)行結(jié)果
2016-02-14-14-51 [pool-2-thread-1] [com.autonavi.task.ScheduledTasks] [INFO] - ScheduledTest.executeUploadBackTask 定時(shí)任務(wù)3:15,name:pool-2-thread-1 定時(shí)任務(wù)2:15 2016-02-14-14-51 [pool-2-thread-1] [com.autonavi.task.ScheduledTasks] [INFO] - ScheduledTest.executeUploadTask 定時(shí)任務(wù)2:15,name:pool-2-thread-1 定時(shí)任務(wù)1:15 2016-02-14-14-52 [pool-2-thread-1] [com.autonavi.task.ScheduledTasks] [INFO] - ScheduledTest.executeFileDownLoadTask 定時(shí)任務(wù)1:15,name:pool-2-thread-1 定時(shí)任務(wù)2:15 2016-02-14-14-52 [pool-2-thread-1] [com.autonavi.task.ScheduledTasks] [INFO] - ScheduledTest.executeUploadTask 定時(shí)任務(wù)2:15,name:pool-2-thread-1 定時(shí)任務(wù)2:15 2016-02-14-14-53 [pool-2-thread-1] [com.autonavi.task.ScheduledTasks] [INFO] - ScheduledTest.executeUploadTask 定時(shí)任務(wù)2:15,name:pool-2-thread-1
5. 串行任務(wù)
上述方法可以實(shí)現(xiàn)定時(shí)任務(wù),方式也比較簡(jiǎn)單,不用配置什么文件啥的,但你會(huì)發(fā)現(xiàn)一個(gè)問題,就是不論定時(shí)任務(wù)被安排在多少個(gè)class類中,其依然是單線程執(zhí)行定時(shí)任務(wù)(串行任務(wù)):
2016-02-14-15-05 [pool-2-thread-1] [com.autonavi.task.ScheduledTasks] [INFO] - ScheduledTasks.executeUploadTask 定時(shí)任務(wù)1:15,name:pool-2-thread-1 定時(shí)任務(wù)2:15 2016-02-14-15-06 [pool-2-thread-1] [com.autonavi.task.ScheduledTasks] [INFO] - ScheduledTest.executeUploadTask 定時(shí)任務(wù)2:15,name:pool-2-thread-1
上述執(zhí)行結(jié)果中ScheduledTest和ScheduledTasks是兩個(gè)獨(dú)立類,都有各自定時(shí)任務(wù),但運(yùn)行時(shí)起Thread Name都是一樣的pool-2-thread-1,因此每個(gè)定時(shí)任務(wù)若要新啟一個(gè)線程,需要自行編寫實(shí)現(xiàn)或者配置文件。
SpringBoot定時(shí)任務(wù)默認(rèn)單線程,多線程需要自行實(shí)現(xiàn)或配置文件
6. 并行任務(wù)
有時(shí)候會(huì)碰到不同業(yè)務(wù)的定時(shí)任務(wù),這時(shí)候利用并行任務(wù)處理要妥當(dāng),采用多線程任務(wù)。只需要配置SpringBoot的配置文件:applicationContext.xml,添加如下內(nèi)容:
<beans xmlns="http://www.springframework.org/schema/beans" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:task="http://www.springframework.org/schema/task" xmlns:context="http://www.springframework.org/schema/context" xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context.xsd http://www.springframework.org/schema/task http://www.springframework.org/schema/task/spring-task-3.0.xsd"> <!-- Enables the Spring Task @Scheduled programming model --> <task:executor id="executor" pool-size="5" /> <task:scheduler id="scheduler" pool-size="10" /> <task:annotation-driven executor="executor" scheduler="scheduler" /> </beans>
添加紅框中的內(nèi)容
同時(shí)注意補(bǔ)充title中遺漏的網(wǎng)址。
效果如下,每個(gè)調(diào)度處理一個(gè)任務(wù),每個(gè)調(diào)度也是一個(gè)子線程:
有關(guān)executor、scheduler參數(shù)的介紹見文中的34.5 The Task Namespace節(jié)。
7. 基于springboot的定時(shí)任務(wù)工程樣例
8. 動(dòng)態(tài)定時(shí)任務(wù)說明
有時(shí)候需要實(shí)現(xiàn)動(dòng)態(tài)定時(shí)任務(wù),即工程啟動(dòng)后,可以實(shí)現(xiàn)啟動(dòng)和關(guān)閉任務(wù),同時(shí)也可以設(shè)置定時(shí)計(jì)劃。這就需要利用到quartz,spring官方對(duì)于這個(gè)包下面各類的介紹,后續(xù)抽空配置下這類業(yè)務(wù)的實(shí)現(xiàn):
http://docs.spring.io/spring/docs/3.2.x/javadoc-api/org/springframework/scheduling/quartz/package-summary.html。
以上就是本文的全部內(nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
java 獲取數(shù)據(jù)庫連接的實(shí)現(xiàn)代碼
本篇文章是對(duì)在java中獲取數(shù)據(jù)庫連接的實(shí)現(xiàn)代碼進(jìn)行了詳細(xì)的分析介紹,需要的朋友參考下2013-05-05Java程序打包成帶參數(shù)的jar文件實(shí)例代碼
這篇文章主要介紹了Java程序打包成帶參數(shù)的jar文件實(shí)例代碼,需要的朋友可以參考下2017-09-09解決Springboot全局異常處理與AOP日志處理中@AfterThrowing失效問題
這篇文章主要介紹了解決Springboot全局異常處理與AOP日志處理中@AfterThrowing失效問題,文中介紹了兩種失效場(chǎng)景,對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)吧2023-05-05java 學(xué)習(xí)筆記(入門篇)_java程序helloWorld
安裝配置完Java的jdk,下面就開始寫第一個(gè)java程序--hello World.用來在控制臺(tái)輸出“Hello World”,接下來詳細(xì)介紹,感興趣的朋友可以參考下2013-01-01idea為java程序添加啟動(dòng)參數(shù)的問題解析(program?arguments,vm?arguments,Envi
這篇文章主要介紹了idea為java程序添加啟動(dòng)參數(shù)的問題解析(program?arguments,vm?arguments,Environment?variable)并在程序中獲取使用,本文給大家分享問題描述及解決方法,需要的朋友可以參考下2023-09-09Java迭代器實(shí)現(xiàn)Python中的range代碼實(shí)例
這篇文章主要介紹了Java迭代器實(shí)現(xiàn)Python中的range代碼實(shí)例,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2020-03-03springboot實(shí)現(xiàn)發(fā)送郵件(QQ郵箱為例)
這篇文章主要為大家詳細(xì)介紹了springboot實(shí)現(xiàn)發(fā)送郵件,qq郵箱代碼實(shí)現(xiàn)郵件發(fā)送,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2020-06-06Spring MVC全局異常處理和單元測(cè)試_動(dòng)力節(jié)點(diǎn)Java學(xué)院整理
本篇文章主要介紹了Spring MVC全局異常處理和單元測(cè)試,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2017-08-08