欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

Springboot自帶定時(shí)任務(wù)實(shí)現(xiàn)動(dòng)態(tài)配置Cron參數(shù)方式

 更新時(shí)間:2021年11月01日 09:58:11   作者:linlangleo  
這篇文章主要介紹了Springboot自帶定時(shí)任務(wù)實(shí)現(xiàn)動(dòng)態(tài)配置Cron參數(shù)方式,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教

Springboot自帶定時(shí)任務(wù)實(shí)現(xiàn)動(dòng)態(tài)配置Cron參數(shù)

同學(xué)們,我今天分享一下SpringBoot動(dòng)態(tài)配置Cron參數(shù)。場(chǎng)景是這樣子的:后臺(tái)管理界面對(duì)定時(shí)任務(wù)進(jìn)行管理,可動(dòng)態(tài)修改執(zhí)行時(shí)間,然后保存入庫(kù),每次任務(wù)執(zhí)行前從庫(kù)里查詢時(shí)間,以達(dá)到動(dòng)態(tài)修改Cron參數(shù)的效果。好,咱們一起來看看是怎么回事。

SpringBoot定時(shí)任務(wù)的四種實(shí)現(xiàn)方式(主要)

  • Timer:這是java自帶的java.util.Timer類,這個(gè)類允許你調(diào)度一個(gè)java.util.TimerTask任務(wù)。使用這種方式可以讓你的程序按照某一個(gè)頻度執(zhí)行,但不能在指定時(shí)間運(yùn)行。一般用的較少。
  • ScheduledExecutorService:也jdk自帶的一個(gè)類;是基于線程池設(shè)計(jì)的定時(shí)任務(wù)類,每個(gè)調(diào)度任務(wù)都會(huì)分配到線程池中的一個(gè)線程去執(zhí)行,也就是說,任務(wù)是并發(fā)執(zhí)行,互不影響。
  • Spring Task:Spring3.0以后自帶的task,可以將它看成一個(gè)輕量級(jí)的Quartz,而且使用起來比Quartz簡(jiǎn)單許多。
  • Quartz:這是一個(gè)功能比較強(qiáng)大的的調(diào)度器,可以讓你的程序在指定時(shí)間執(zhí)行,也可以按照某一個(gè)頻度執(zhí)行,配置起來稍顯復(fù)雜。

1.1使用Timer

這是讓你按照固定的頻率去執(zhí)行一個(gè)任務(wù),不能指定時(shí)間。

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ù)在指定的時(shí)間開始進(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í)行的延時(shí)時(shí)間
        //      3、任務(wù)執(zhí)行間隔 4、間隔時(shí)間單位
        service.scheduleAtFixedRate(()->System.out.println("task ScheduledExecutorService "+new Date()), 0, 3, TimeUnit.SECONDS);
    }
}

1.3使用Spring Task

我們主要講解它的動(dòng)態(tài)配置使用方法。

在剛開始使用的時(shí)候,我們更改一個(gè)任務(wù)的執(zhí)行時(shí)間,一般是這樣的:修改定時(shí)任務(wù)的執(zhí)行周期,把服務(wù)停下來,改下任務(wù)的cron參數(shù),再重啟服務(wù)就搞搞定了。這種方式很簡(jiǎn)單,沒有可說的,但是有沒有一種可能,在不停服務(wù)的情況下,就可以動(dòng)態(tài)的修改任務(wù)的cron參數(shù)呢?那是必須有!

剛剛提到的方法里,我們?cè)谥黝惿厦婕覢EnableScheduling注解,在任務(wù)方法前面加上@Scheduled(cron =“0/5 * * * * *”)注解定義執(zhí)行時(shí)間,但是動(dòng)態(tài)配置的步驟就有點(diǎn)不一樣:

1. 在定時(shí)任務(wù)類上增加@EnabledScheduling注解,并實(shí)現(xiàn)SchedulingConfigurer接口。

2. 設(shè)置一個(gè)靜態(tài)的cron,用于存放任務(wù)執(zhí)行周期參數(shù)。

3. 從數(shù)據(jù)庫(kù)獲取Cron參數(shù),用于模擬實(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) {
        //項(xiàng)目部署時(shí),會(huì)在這里執(zhí)行一次,從數(shù)據(jù)庫(kù)拿到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ā),都會(huì)執(zhí)行這里的方法一次,重新獲取下一次的執(zhí)行時(shí)間        
              cron = timerQueryMapper.getCronTime();
              CronTrigger trigger = new CronTrigger(cron);
              Date nextExec = trigger.nextExecutionTime(triggerContext);
              return nextExec;
           }
       };
       taskRegistrar.addTriggerTask(task, trigger);
    }
}

因?yàn)槭且蝿?wù)執(zhí)行一次的時(shí)候才會(huì)去修改時(shí)間的cron表達(dá)式,所以改了cron后,要在下下次任務(wù)執(zhí)行時(shí)才會(huì)生效。

這里核心的主要是使用到了ScheduledTaskRegistrar這個(gè)類有一個(gè)方法addTriggerTask(Runnable,Trigger) 兩個(gè)參數(shù),一個(gè)Runnable,一個(gè)是Trigger,在Runnable中執(zhí)行業(yè)務(wù)邏輯代碼,在Trigger修改定時(shí)任務(wù)的執(zhí)行周期。

1.4整合Quartz

在SpringBoot版本是2.0.0以后的,則在spring-boot-starter中已經(jīng)包含了quart的依賴,則可以直接使用spring-boot-starter-quartz依賴,如果是低于2.0.0版本的,需要額外添加quartz的依賴。

spring動(dòng)態(tài)配置cron表達(dá)式,不需要停服

spring做定時(shí)任務(wù)調(diào)度時(shí)有常用的兩種方式,分別是基于配置文件的quartz和基于注解的@Scheduler。

quartz需要較多的配置文件,個(gè)人感覺比較麻煩,@Scheduler注解只需要簡(jiǎn)單的配置即可,但是這兩種發(fā)方法不能動(dòng)態(tài)加載cron表達(dá)式,每次更改調(diào)度規(guī)則都需要重啟服務(wù)。

本文介紹一種不需要重啟服務(wù)的動(dòng)態(tài)加載cron表達(dá)式的方法。

SchedulingConfigurer接口實(shí)現(xiàn)動(dòng)態(tài)加載cron表達(dá)式

代碼示例如下:

@Component
@EnableScheduling
public class Test implements SchedulingConfigurer {
 
    @Override
    public void configureTasks(ScheduledTaskRegistrar scheduledTaskRegistrar) {
        //創(chuàng)建一個(gè)線程池調(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)度時(shí)加載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 * * * * *

運(yùn)行結(jié)果(為了查看方便,只運(yùn)行一個(gè)任務(wù)):

以上為個(gè)人經(jīng)驗(yàn),希望能給大家一個(gè)參考,也希望大家多多支持腳本之家。

相關(guān)文章

  • SpringBoot+Echarts實(shí)現(xiàn)請(qǐng)求后臺(tái)數(shù)據(jù)顯示餅狀圖

    SpringBoot+Echarts實(shí)現(xiàn)請(qǐng)求后臺(tái)數(shù)據(jù)顯示餅狀圖

    這篇文章主要介紹了SpringBoot+Echarts實(shí)現(xiàn)請(qǐng)求后臺(tái)數(shù)據(jù)顯示餅狀圖,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2019-12-12
  • SpringBoot2.x設(shè)置Session失效時(shí)間及失效跳轉(zhuǎn)方式

    SpringBoot2.x設(shè)置Session失效時(shí)間及失效跳轉(zhuǎn)方式

    這篇文章主要介紹了SpringBoot2.x設(shè)置Session失效時(shí)間及失效跳轉(zhuǎn)方式,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2022-03-03
  • 深入了解final在java中的應(yīng)用

    深入了解final在java中的應(yīng)用

    談到final關(guān)鍵字,想必很多人都不陌生,在使用匿名內(nèi)部類的時(shí)候可能會(huì)經(jīng)常用到final關(guān)鍵字。另外,Java中的String類就是一個(gè)final類,那么今天我們就來了解final這個(gè)關(guān)鍵字的用法。
    2019-06-06
  • httpclient connect連接請(qǐng)求方法源碼解讀

    httpclient connect連接請(qǐng)求方法源碼解讀

    這篇文章主要為大家介紹了httpclient connect連接請(qǐng)求方法解讀,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪
    2023-11-11
  • Java lambda表達(dá)式實(shí)現(xiàn)Flink WordCount過程解析

    Java lambda表達(dá)式實(shí)現(xiàn)Flink WordCount過程解析

    這篇文章主要介紹了Java lambda表達(dá)式實(shí)現(xiàn)Flink WordCount過程解析,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下
    2020-02-02
  • 深入理解Hibernate中的flush機(jī)制

    深入理解Hibernate中的flush機(jī)制

    這篇文章主要介紹了深入理解Hibernate中的flush機(jī)制,本文是對(duì)flush機(jī)制深入研究得出的一些結(jié)論總結(jié),需要的朋友可以參考下
    2015-01-01
  • javaSE中異常如何處理舉例詳解

    javaSE中異常如何處理舉例詳解

    程序運(yùn)行過程中發(fā)生了不正常的情況,這種不正常的情況叫做異常,下面這篇文章主要給大家介紹了關(guān)于javaSE中異常如何處理的相關(guān)資料,文中通過實(shí)例代碼介紹的非常詳細(xì),需要的朋友可以參考下
    2023-01-01
  • 解決springmvc使用@PathVariable路徑匹配問題

    解決springmvc使用@PathVariable路徑匹配問題

    這篇文章主要介紹了解決springmvc使用@PathVariable路徑匹配問題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過來看看吧
    2021-02-02
  • 全面匯總SpringBoot和SpringClould常用注解

    全面匯總SpringBoot和SpringClould常用注解

    Java注解是附加在代碼中的一些元信息,用于一些工具在編譯、運(yùn)行時(shí)進(jìn)行解析和使用,起到說明、配置的功能,這篇文章就帶你來了解一下
    2021-08-08
  • 詳解Java多線程編程中CountDownLatch阻塞線程的方法

    詳解Java多線程編程中CountDownLatch阻塞線程的方法

    在Java中和ReadWriteLock.ReadLock一樣,CountDownLatch的本質(zhì)也是一個(gè)"共享鎖",這里我們就來詳解Java多線程編程中CountDownLatch阻塞線程的方法:
    2016-07-07

最新評(píng)論