Springboot自帶定時任務(wù)實現(xiàn)動態(tài)配置Cron參數(shù)方式
Springboot自帶定時任務(wù)實現(xiàn)動態(tài)配置Cron參數(shù)
同學(xué)們,我今天分享一下SpringBoot動態(tài)配置Cron參數(shù)。場景是這樣子的:后臺管理界面對定時任務(wù)進(jìn)行管理,可動態(tài)修改執(zhí)行時間,然后保存入庫,每次任務(wù)執(zhí)行前從庫里查詢時間,以達(dá)到動態(tài)修改Cron參數(shù)的效果。好,咱們一起來看看是怎么回事。
SpringBoot定時任務(wù)的四種實現(xiàn)方式(主要)
Timer:這是java自帶的java.util.Timer類,這個類允許你調(diào)度一個java.util.TimerTask任務(wù)。使用這種方式可以讓你的程序按照某一個頻度執(zhí)行,但不能在指定時間運行。一般用的較少。ScheduledExecutorService:也jdk自帶的一個類;是基于線程池設(shè)計的定時任務(wù)類,每個調(diào)度任務(wù)都會分配到線程池中的一個線程去執(zhí)行,也就是說,任務(wù)是并發(fā)執(zhí)行,互不影響。Spring Task:Spring3.0以后自帶的task,可以將它看成一個輕量級的Quartz,而且使用起來比Quartz簡單許多。Quartz:這是一個功能比較強大的的調(diào)度器,可以讓你的程序在指定時間執(zhí)行,也可以按照某一個頻度執(zhí)行,配置起來稍顯復(fù)雜。
1.1使用Timer
這是讓你按照固定的頻率去執(zhí)行一個任務(wù),不能指定時間。
public class TestTimer {
public static void main(String[] args) {
TimerTask timerTask = new TimerTask() {
@Override
public void run() {
System.out.println("task run:"+ new Date());
}
};
Timer timer = new Timer();
//安排指定的任務(wù)在指定的時間開始進(jìn)行重復(fù)的固定延遲執(zhí)行。這里是每3秒執(zhí)行一次
timer.schedule(timerTask,10,3000);
}
}1.2使用ScheduledExecutorService和timer類似
public class TestScheduledExecutorService {
public static void main(String[] args) {
ScheduledExecutorService service = Executors.newSingleThreadScheduledExecutor();
// 參數(shù):1、任務(wù)體 2、首次執(zhí)行的延時時間
// 3、任務(wù)執(zhí)行間隔 4、間隔時間單位
service.scheduleAtFixedRate(()->System.out.println("task ScheduledExecutorService "+new Date()), 0, 3, TimeUnit.SECONDS);
}
}1.3使用Spring Task
我們主要講解它的動態(tài)配置使用方法。
在剛開始使用的時候,我們更改一個任務(wù)的執(zhí)行時間,一般是這樣的:修改定時任務(wù)的執(zhí)行周期,把服務(wù)停下來,改下任務(wù)的cron參數(shù),再重啟服務(wù)就搞搞定了。這種方式很簡單,沒有可說的,但是有沒有一種可能,在不停服務(wù)的情況下,就可以動態(tài)的修改任務(wù)的cron參數(shù)呢?那是必須有!
剛剛提到的方法里,我們在主類上面加@EnableScheduling注解,在任務(wù)方法前面加上@Scheduled(cron =“0/5 * * * * *”)注解定義執(zhí)行時間,但是動態(tài)配置的步驟就有點不一樣:
1. 在定時任務(wù)類上增加@EnabledScheduling注解,并實現(xiàn)SchedulingConfigurer接口。
2. 設(shè)置一個靜態(tài)的cron,用于存放任務(wù)執(zhí)行周期參數(shù)。
3. 從數(shù)據(jù)庫獲取Cron參數(shù),用于模擬實際業(yè)務(wù)中外部原因修改了任務(wù)執(zhí)行周期。
4. 設(shè)置任務(wù)觸發(fā)器,觸發(fā)任務(wù)執(zhí)行。
import java.util.Date;
import org.springframework.scheduling.Trigger;
import org.springframework.scheduling.TriggerContext;
import org.springframework.scheduling.annotation.EnableScheduling;
import org.springframework.scheduling.annotation.SchedulingConfigurer;
import org.springframework.scheduling.config.ScheduledTaskRegistrar;
import org.springframework.scheduling.support.CronTrigger;
import org.springframework.stereotype.Component;
import java.time.LocalDateTime;
@Component
@EnableScheduling
public class TaskCronChange implements SchedulingConfigurer{
public static String cron;
@Override
public void configureTasks(ScheduledTaskRegistrar taskRegistrar) {
//項目部署時,會在這里執(zhí)行一次,從數(shù)據(jù)庫拿到cron表達(dá)式
cron = timerQueryMapper.getCronTime();
Runnable task = new Runnable() {
@Override
public void run() {
//任務(wù)邏輯代碼部分.
System.out.println("I am going:" + LocalDateTime.now());
}
};
Trigger trigger = new Trigger() {
@Override
public Date nextExecutionTime(TriggerContext triggerContext) {
//任務(wù)觸發(fā),可修改任務(wù)的執(zhí)行周期.
//每一次任務(wù)觸發(fā),都會執(zhí)行這里的方法一次,重新獲取下一次的執(zhí)行時間
cron = timerQueryMapper.getCronTime();
CronTrigger trigger = new CronTrigger(cron);
Date nextExec = trigger.nextExecutionTime(triggerContext);
return nextExec;
}
};
taskRegistrar.addTriggerTask(task, trigger);
}
}
因為是要任務(wù)執(zhí)行一次的時候才會去修改時間的cron表達(dá)式,所以改了cron后,要在下下次任務(wù)執(zhí)行時才會生效。
這里核心的主要是使用到了ScheduledTaskRegistrar這個類有一個方法addTriggerTask(Runnable,Trigger) 兩個參數(shù),一個Runnable,一個是Trigger,在Runnable中執(zhí)行業(yè)務(wù)邏輯代碼,在Trigger修改定時任務(wù)的執(zhí)行周期。
1.4整合Quartz
在SpringBoot版本是2.0.0以后的,則在spring-boot-starter中已經(jīng)包含了quart的依賴,則可以直接使用spring-boot-starter-quartz依賴,如果是低于2.0.0版本的,需要額外添加quartz的依賴。
spring動態(tài)配置cron表達(dá)式,不需要停服
spring做定時任務(wù)調(diào)度時有常用的兩種方式,分別是基于配置文件的quartz和基于注解的@Scheduler。
quartz需要較多的配置文件,個人感覺比較麻煩,@Scheduler注解只需要簡單的配置即可,但是這兩種發(fā)方法不能動態(tài)加載cron表達(dá)式,每次更改調(diào)度規(guī)則都需要重啟服務(wù)。
本文介紹一種不需要重啟服務(wù)的動態(tài)加載cron表達(dá)式的方法。
SchedulingConfigurer接口實現(xiàn)動態(tài)加載cron表達(dá)式
代碼示例如下:
@Component
@EnableScheduling
public class Test implements SchedulingConfigurer {
@Override
public void configureTasks(ScheduledTaskRegistrar scheduledTaskRegistrar) {
//創(chuàng)建一個線程池調(diào)度器,默認(rèn)是單線程執(zhí)行
ScheduledExecutorService executorService = Executors.newScheduledThreadPool(100);
scheduledTaskRegistrar.setScheduler(executorService);
//增加任務(wù)
scheduledTaskRegistrar.addTriggerTask(new Task("test1"),new Trig("cronExpess1"));
scheduledTaskRegistrar.addTriggerTask(new Task("test2"),new Trig("cronExpess2"));
scheduledTaskRegistrar.addTriggerTask(new Task("test3"),new Trig("cronExpess2"));
}
}
/**
* 業(yè)務(wù)類
*/
class Task implements Runnable{
String task;
public Task(String task){
this.task = task;
}
//具體業(yè)務(wù)
@Override
public void run() {
System.out.println(task+":"+LocalDateTime.now()+","+Thread.currentThread().getName());
}
}
/**
* 調(diào)度類
*/
class Trig implements Trigger{
private String cronExpress;
public Trig(String cronExpress){
this.cronExpress = cronExpress;
}
@Override
public Date nextExecutionTime(TriggerContext triggerContext) {
String cron = null;
try {
//每次調(diào)度時加載cron表達(dá)式
cron = new Config().getCrons().get(cronExpress);
} catch (IOException e) {
e.printStackTrace();
}
CronTrigger cronTrigger = new CronTrigger(cron);
return cronTrigger.nextExecutionTime(triggerContext);
}
}
/**
* 加載cron表達(dá)式
*/
class Config{
private static Map<String,String> cronMap;
private static long preModifyTime;
private String cronFile = "config/application.properties";
public Map<String,String> getCrons() throws IOException {
File file = new File(cronFile);
long nowModifyTime = file.lastModified();
if (cronMap != null && nowModifyTime == preModifyTime){
return cronMap;
}else {
cronMap = new HashMap<>();
BufferedReader br = new BufferedReader(new FileReader(file));
String line = null;
while ((line = br.readLine()) != null){
String[] s = line.split("=");
cronMap.put(s[0].trim(),s[1].trim());
}
preModifyTime = nowModifyTime;
return cronMap;
}
}
}配置文件:
cronExpess1 = 0/5 * * * * * cronExpess2 = 0/10 * * * * *
運行結(jié)果(為了查看方便,只運行一個任務(wù)):

以上為個人經(jīng)驗,希望能給大家一個參考,也希望大家多多支持腳本之家。
- SpringBoot設(shè)置動態(tài)定時任務(wù)的方法詳解
- 淺談SpringBoot集成Quartz動態(tài)定時任務(wù)
- 基于Springboot執(zhí)行多個定時任務(wù)并動態(tài)獲取定時任務(wù)信息
- SpringBoot實現(xiàn)動態(tài)定時任務(wù)
- Springboot整個Quartz實現(xiàn)動態(tài)定時任務(wù)的示例代碼
- 詳解SpringBoot 創(chuàng)建定時任務(wù)(配合數(shù)據(jù)庫動態(tài)執(zhí)行)
- springboot整合Quartz實現(xiàn)動態(tài)配置定時任務(wù)的方法
- SpringBoot實現(xiàn)動態(tài)增刪啟停定時任務(wù)的方式
相關(guān)文章
SpringBoot+Echarts實現(xiàn)請求后臺數(shù)據(jù)顯示餅狀圖
這篇文章主要介紹了SpringBoot+Echarts實現(xiàn)請求后臺數(shù)據(jù)顯示餅狀圖,文中示例代碼介紹的非常詳細(xì),具有一定的參考價值,感興趣的小伙伴們可以參考一下2019-12-12
SpringBoot2.x設(shè)置Session失效時間及失效跳轉(zhuǎn)方式
這篇文章主要介紹了SpringBoot2.x設(shè)置Session失效時間及失效跳轉(zhuǎn)方式,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2022-03-03
Java lambda表達(dá)式實現(xiàn)Flink WordCount過程解析
這篇文章主要介紹了Java lambda表達(dá)式實現(xiàn)Flink WordCount過程解析,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友可以參考下2020-02-02
解決springmvc使用@PathVariable路徑匹配問題
這篇文章主要介紹了解決springmvc使用@PathVariable路徑匹配問題,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧2021-02-02
全面匯總SpringBoot和SpringClould常用注解
Java注解是附加在代碼中的一些元信息,用于一些工具在編譯、運行時進(jìn)行解析和使用,起到說明、配置的功能,這篇文章就帶你來了解一下2021-08-08
詳解Java多線程編程中CountDownLatch阻塞線程的方法
在Java中和ReadWriteLock.ReadLock一樣,CountDownLatch的本質(zhì)也是一個"共享鎖",這里我們就來詳解Java多線程編程中CountDownLatch阻塞線程的方法:2016-07-07

