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

SpringBoot的@Scheduled和@Schedules區(qū)別小結(jié)

 更新時(shí)間:2025年01月22日 10:32:23   作者:山高自有客行路  
本文主要介紹了SpringBoot的@Scheduled和@Schedules區(qū)別小結(jié),文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧

@Scheduled 的詳細(xì)解析

參數(shù)詳解

  • cron: 使用Cron表達(dá)式來(lái)指定復(fù)雜的調(diào)度模式。Cron表達(dá)式的格式如下:

    • 秒(0-59)
    • 分鐘(0-59)
    • 小時(shí)(0-23)
    • 日(1-31)
    • 月(1-12 或 JAN-DEC)
    • 星期(0-7 或 SUN-SAT,其中0和7都表示星期日)
    • 年(可選,1970-2099)

    Cron表達(dá)式的每個(gè)字段可以是具體的值、范圍、列表或通配符(*)。例如:

    • "0 0 12 * * ?" 表示每天中午12點(diǎn)。
    • "0 15 10 ? * MON-FRI" 表示周一至周五上午10:15執(zhí)行。
    • "0 0/5 * * * ?" 表示每5分鐘執(zhí)行一次。
    • "0 0 12 1 * ?" 表示每月第一天中午12點(diǎn)執(zhí)行。
  • fixedRate: 指定以固定的速率重復(fù)執(zhí)行任務(wù),從前一次任務(wù)開始時(shí)刻算起。它不會(huì)等待前一個(gè)任務(wù)完成,因此如果任務(wù)執(zhí)行時(shí)間超過(guò)了設(shè)定的時(shí)間間隔,可能會(huì)有重疊的任務(wù)實(shí)例在運(yùn)行。

  • fixedDelay: 類似于 fixedRate,但是它是以前一次任務(wù)的完成時(shí)刻作為下一次任務(wù)啟動(dòng)的時(shí)間基準(zhǔn)。這種方式可以確保每次只有一個(gè)任務(wù)實(shí)例在運(yùn)行,前提是任務(wù)的執(zhí)行時(shí)間短于延遲時(shí)間。

  • initialDelay: 在第一次執(zhí)行之前等待的時(shí)間(毫秒)。這個(gè)參數(shù)通常與 fixedRate 或 fixedDelay 一起使用,用來(lái)設(shè)置首次執(zhí)行前的延遲。

  • zone: 定義時(shí)區(qū),默認(rèn)是系統(tǒng)的默認(rèn)時(shí)區(qū)。如果你的應(yīng)用需要在全球不同地區(qū)運(yùn)行,明確指定時(shí)區(qū)可能是很重要的。

Cron表達(dá)式、fixedRate、fixedDelay、initialDelay如何選擇     

  • 如果你需要非常具體的調(diào)度模式,如每天凌晨?jī)牲c(diǎn)執(zhí)行某個(gè)任務(wù),那么應(yīng)該選擇Cron表達(dá)式。
  • 如果你希望任務(wù)以固定的速率重復(fù)執(zhí)行,不論每次執(zhí)行花費(fèi)多少時(shí)間,你應(yīng)該選擇fixedRate
  • 如果你希望在前一個(gè)任務(wù)完全結(jié)束后再等待一段固定時(shí)間才開始下一個(gè)任務(wù),那么fixedDelay是更好的選擇。
  • 如果你需要設(shè)置首次執(zhí)行的延遲,可以添加initialDelay參數(shù)到你的調(diào)度配置中。    

示例代碼

import org.springframework.scheduling.annotation.Scheduled;
import org.springframework.stereotype.Component;

@Component
public class ScheduledTasks {

    // 每天中午12點(diǎn)執(zhí)行(上海時(shí)區(qū))
    @Scheduled(cron = "0 0 12 * * ?", zone = "Asia/Shanghai")
    public void scheduledTaskUsingCron() {
        System.out.println("Scheduled task using cron at Asia/Shanghai timezone.");
    }

    // 每5秒執(zhí)行一次,首次執(zhí)行前等待2秒
    @Scheduled(fixedRate = 5000, initialDelay = 2000)
    public void scheduledTaskWithFixedRate() {
        System.out.println("Scheduled task with fixed rate.");
    }

    // 上次任務(wù)完成后等待3秒再執(zhí)行下一次
    @Scheduled(fixedDelay = 3000)
    public void scheduledTaskWithFixedDelay() {
        System.out.println("Scheduled task with fixed delay.");
    }
}

@Schedules 的詳細(xì)解析

@Schedules 允許多個(gè) @Scheduled 注解組合在一起,為同一個(gè)方法設(shè)定多種不同的調(diào)度策略。這對(duì)于那些需要在多個(gè)不同時(shí)間點(diǎn)或條件下觸發(fā)的方法非常有用。

示例代碼

import org.springframework.scheduling.annotation.Schedules;
import org.springframework.scheduling.annotation.Scheduled;
import org.springframework.stereotype.Component;

@Component
public class MultipleScheduledTasks {

    // 每天中午12點(diǎn)執(zhí)行,并且每5秒也執(zhí)行一次
    @Schedules({
        @Scheduled(cron = "0 0 12 * * ?"),
        @Scheduled(fixedRate = 5000)
    })
    public void multipleScheduledTasks() {
        System.out.println("Multiple scheduled tasks.");
    }
}

啟用和管理定時(shí)任務(wù)

要使這些注解生效,你需要確保你的Spring應(yīng)用已經(jīng)啟用了對(duì)它們的支持。這可以通過(guò)在配置類上添加 @EnableScheduling 來(lái)實(shí)現(xiàn):

import org.springframework.context.annotation.Configuration;
import org.springframework.scheduling.annotation.EnableScheduling;

@Configuration
@EnableScheduling
public class SchedulingConfig {
    // 配置類內(nèi)容
}

自定義 TaskScheduler

對(duì)于更復(fù)雜的需求,比如調(diào)整線程池大小或者設(shè)置線程名稱前綴等,你可以通過(guò)自定義 TaskScheduler 來(lái)進(jìn)行配置。Spring提供了幾種內(nèi)置的調(diào)度器實(shí)現(xiàn),如 ThreadPoolTaskScheduler 和 ConcurrentTaskScheduler。

示例代碼

import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.scheduling.concurrent.ThreadPoolTaskScheduler;

@Configuration
public class SchedulerConfig {

    @Bean
    public ThreadPoolTaskScheduler taskScheduler() {
        ThreadPoolTaskScheduler taskScheduler = new ThreadPoolTaskScheduler();
        taskScheduler.setPoolSize(10); // 設(shè)置線程池大小
        taskScheduler.setThreadNamePrefix("MyScheduledTask-");
        taskScheduler.setErrorHandler(t -> {
            System.err.println("Error occurred in scheduled task: " + t.getMessage());
        });
        taskScheduler.setWaitForTasksToCompleteOnShutdown(true);
        taskScheduler.setAwaitTerminationSeconds(60);
        return taskScheduler;
    }
}

錯(cuò)誤處理

當(dāng)一個(gè)預(yù)定任務(wù)拋出異常時(shí),默認(rèn)情況下Spring會(huì)記錄錯(cuò)誤日志,但任務(wù)本身不會(huì)被取消。如果你想改變這種行為,可以使用 DelegatingErrorHandlingRunnable 或者直接在 ThreadPoolTaskScheduler 中設(shè)置錯(cuò)誤處理器(如上面的示例所示)。

自定義錯(cuò)誤處理邏輯

你可以創(chuàng)建自己的錯(cuò)誤處理器來(lái)捕獲并處理異常:

import org.springframework.aop.interceptor.SimpleAsyncUncaughtExceptionHandler;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.scheduling.annotation.AsyncConfigurer;
import org.springframework.scheduling.concurrent.ThreadPoolTaskExecutor;

import java.util.concurrent.Executor;

@Configuration
public class AsyncConfig implements AsyncConfigurer {

    @Override
    public Executor getAsyncExecutor() {
        ThreadPoolTaskExecutor executor = new ThreadPoolTaskExecutor();
        executor.setCorePoolSize(2);
        executor.setMaxPoolSize(5);
        executor.setQueueCapacity(100);
        executor.setThreadNamePrefix("Async-");
        executor.initialize();
        return executor;
    }

    @Override
    public AsyncUncaughtExceptionHandler getAsyncUncaughtExceptionHandler() {
        return new SimpleAsyncUncaughtExceptionHandler() {
            @Override
            public void handleUncaughtException(Throwable ex, Method method, Object... params) {
                // 自定義異常處理邏輯
                System.err.println("Exception in async task: " + ex.getMessage());
            }
        };
    }
}

最佳實(shí)踐

  • 避免長(zhǎng)時(shí)間運(yùn)行的任務(wù):盡量不要讓預(yù)定任務(wù)執(zhí)行過(guò)長(zhǎng)的時(shí)間,因?yàn)樗鼈兛赡軙?huì)阻塞其他任務(wù)的執(zhí)行。如果任務(wù)有可能運(yùn)行很長(zhǎng)時(shí)間,請(qǐng)考慮將其拆分為更小的部分,或者使用異步處理。

  • 任務(wù)沖突管理:當(dāng)使用 fixedRate 時(shí),要注意任務(wù)可能會(huì)重疊。如果任務(wù)執(zhí)行時(shí)間可能超過(guò)間隔時(shí)間,應(yīng)該選擇 fixedDelay 來(lái)避免這種情況。

  • 資源清理:確保在任務(wù)結(jié)束時(shí)正確釋放任何獲取的資源,比如數(shù)據(jù)庫(kù)連接或文件句柄。

  • 監(jiān)控和報(bào)警:建立適當(dāng)?shù)谋O(jiān)控和報(bào)警機(jī)制,以便在任務(wù)失敗時(shí)能夠及時(shí)收到通知并采取行動(dòng)??梢岳肧pring Boot Actuator提供的健康檢查端點(diǎn),或者集成第三方監(jiān)控工具如Prometheus、Grafana等。

  • 冪等性設(shè)計(jì):確保任務(wù)邏輯具有冪等性,即多次執(zhí)行相同的任務(wù)不會(huì)導(dǎo)致不一致的結(jié)果。這在分布式環(huán)境中尤為重要。

  • 日志記錄:為每個(gè)任務(wù)添加詳細(xì)的日志記錄,包括任務(wù)開始時(shí)間和結(jié)束時(shí)間,以便追蹤任務(wù)執(zhí)行情況。

  • 測(cè)試:編寫單元測(cè)試和集成測(cè)試來(lái)驗(yàn)證定時(shí)任務(wù)的行為是否符合預(yù)期。可以使用Mockito或其他測(cè)試框架模擬依賴服務(wù)。

  • 多實(shí)例部署的問(wèn)題:在多實(shí)例部署的情況下,所有的實(shí)例都會(huì)嘗試執(zhí)行相同的定時(shí)任務(wù),這可能導(dǎo)致數(shù)據(jù)競(jìng)爭(zhēng)或重復(fù)執(zhí)行。一種解決方案是使用分布式鎖,如Redisson提供的RedLock,來(lái)保證同一時(shí)間只有一個(gè)實(shí)例執(zhí)行特定的任務(wù)。

  • 性能優(yōu)化:對(duì)于高并發(fā)場(chǎng)景下的定時(shí)任務(wù),應(yīng)該評(píng)估線程池的大小和任務(wù)的執(zhí)行頻率,避免因過(guò)多的任務(wù)同時(shí)啟動(dòng)而導(dǎo)致資源耗盡??梢酝ㄟ^(guò)限流、隊(duì)列管理和異步處理等方式提高系統(tǒng)的穩(wěn)定性和響應(yīng)速度。

處理定時(shí)任務(wù)中的常見問(wèn)題

  • 任務(wù)未按預(yù)期執(zhí)行:檢查日志以確定是否有任何異?;蝈e(cuò)誤信息。確保任務(wù)方法是非靜態(tài)的,并且沒(méi)有被final修飾。確認(rèn) @EnableScheduling 已經(jīng)正確啟用。另外,檢查是否存在其他因素阻止任務(wù)執(zhí)行,如網(wǎng)絡(luò)延遲或依賴服務(wù)不可用。

  • 任務(wù)執(zhí)行順序混亂:如果你有多個(gè)任務(wù)幾乎同時(shí)執(zhí)行,可能會(huì)出現(xiàn)執(zhí)行順序混亂的情況。確保你理解了 fixedRate 和 fixedDelay 的區(qū)別,并根據(jù)需要選擇合適的方式。此外,可以通過(guò)增加任務(wù)之間的最小間隔時(shí)間來(lái)減少?zèng)_突的可能性。

  • 多實(shí)例部署的問(wèn)題:在多實(shí)例部署的情況下,所有實(shí)例都會(huì)嘗試執(zhí)行相同的定時(shí)任務(wù)。為了解決這個(gè)問(wèn)題,可以引入分布式鎖機(jī)制,如基于Redis的鎖或Zookeeper的臨時(shí)節(jié)點(diǎn),以確保同一時(shí)間只有一個(gè)實(shí)例執(zhí)行任務(wù)。

  • 長(zhǎng)時(shí)間運(yùn)行的任務(wù):盡量不要讓預(yù)定任務(wù)執(zhí)行過(guò)長(zhǎng)的時(shí)間,因?yàn)樗鼈兛赡軙?huì)阻塞其他任務(wù)的執(zhí)行。如果任務(wù)有可能運(yùn)行很長(zhǎng)時(shí)間,請(qǐng)考慮將其拆分為更小的部分,或者使用異步處理,如通過(guò)消息隊(duì)列(MQ)來(lái)分發(fā)任務(wù)。

  • 時(shí)區(qū)問(wèn)題:確保你的應(yīng)用程序正確處理時(shí)區(qū)差異,特別是在全球范圍內(nèi)運(yùn)行時(shí)。可以在 @Scheduled 注解中顯式指定 zone 參數(shù),或者在整個(gè)應(yīng)用程序中統(tǒng)一配置默認(rèn)時(shí)區(qū)。

案例分析

假設(shè)你正在開發(fā)一個(gè)電子商務(wù)平臺(tái),需要每天凌晨2點(diǎn)生成前一天的銷售報(bào)告。你可以使用 @Scheduled 注解來(lái)安排這個(gè)任務(wù):

import org.springframework.scheduling.annotation.Scheduled;
import org.springframework.stereotype.Service;

@Service
public class DailyReportService {

    @Scheduled(cron = "0 0 2 * * ?", zone = "Asia/Shanghai")
    public void generateDailySalesReport() {
        // 執(zhí)行生成銷售報(bào)告的邏輯
        System.out.println("Generating daily sales report at Asia/Shanghai timezone.");
    }
}

此外,你還可以結(jié)合上述的最佳實(shí)踐來(lái)增強(qiáng)任務(wù)的可靠性,例如:

  • 確保任務(wù)具有冪等性,即使由于某種原因任務(wù)重復(fù)執(zhí)行也不會(huì)影響結(jié)果。
  • 添加詳細(xì)的日志記錄,幫助追蹤任務(wù)的執(zhí)行情況。
  • 實(shí)現(xiàn)錯(cuò)誤處理邏輯,確保即使發(fā)生異常也能得到妥善處理。
  • 如果平臺(tái)有多實(shí)例部署,考慮使用分布式鎖來(lái)防止多個(gè)實(shí)例同時(shí)生成報(bào)告。

 到此這篇關(guān)于SpringBoot的@Scheduled和@Schedules區(qū)別小結(jié)的文章就介紹到這了,更多相關(guān)SpringBoot @Scheduled和@Schedules內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • SpringBoot使用自動(dòng)配置xxxAutoConfiguration

    SpringBoot使用自動(dòng)配置xxxAutoConfiguration

    這篇文章介紹了SpringBoot自動(dòng)配置xxxAutoConfiguration的使用方法,對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧
    2021-12-12
  • Java非靜態(tài)成員變量之死循環(huán)(詳解)

    Java非靜態(tài)成員變量之死循環(huán)(詳解)

    下面小編就為大家?guī)?lái)一篇Java非靜態(tài)成員變量之死循環(huán)(詳解)。小編覺(jué)得挺不錯(cuò)的,現(xiàn)在就分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧
    2017-09-09
  • 淺析Java單例設(shè)計(jì)模式(自寫demo)

    淺析Java單例設(shè)計(jì)模式(自寫demo)

    Java單例模式是看起來(lái)以及用起來(lái)簡(jiǎn)單的一種設(shè)計(jì)模式,但是就實(shí)現(xiàn)方式以及原理來(lái)說(shuō),也并不淺顯,下面這篇文章主要給大家詳細(xì)介紹了Java中單例模式,需要的朋友可以參考下
    2021-12-12
  • SpringBoot整合Junit實(shí)例過(guò)程解析

    SpringBoot整合Junit實(shí)例過(guò)程解析

    這篇文章主要介紹了SpringBoot整合Junit實(shí)例過(guò)程解析,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下
    2019-11-11
  • 必須要學(xué)會(huì)的JMM與volatile

    必須要學(xué)會(huì)的JMM與volatile

    這篇文章主要介紹了必須要學(xué)會(huì)的JMM與volatile,文章圍繞主題展開詳細(xì)的內(nèi)容介紹,具有一定的參考價(jià)值,需要的小伙伴可以參考一下
    2022-09-09
  • SpringBoot2零基礎(chǔ)到精通之映射與常用注解請(qǐng)求處理

    SpringBoot2零基礎(chǔ)到精通之映射與常用注解請(qǐng)求處理

    SpringBoot是一種整合Spring技術(shù)棧的方式(或者說(shuō)是框架),同時(shí)也是簡(jiǎn)化Spring的一種快速開發(fā)的腳手架,本篇讓我們一起學(xué)習(xí)映射、常用注解和方法參數(shù)的小技巧
    2022-03-03
  • java?http請(qǐng)求設(shè)置代理Proxy的兩種常見方法

    java?http請(qǐng)求設(shè)置代理Proxy的兩種常見方法

    代理是一種常見的設(shè)計(jì)模式,其目的就是為其他對(duì)象提供一個(gè)代理以控制對(duì)某個(gè)對(duì)象的訪問(wèn),這篇文章主要給大家介紹了關(guān)于java?http請(qǐng)求設(shè)置代理Proxy的兩種常見方法,需要的朋友可以參考下
    2023-11-11
  • httpclient staleConnectionCheckEnabled獲取連接流程解析

    httpclient staleConnectionCheckEnabled獲取連接流程解析

    這篇文章主要為大家介紹了httpclient staleConnectionCheckEnabled獲取連接流程示例解析,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪
    2023-11-11
  • 使用ftpClient下載ftp上所有文件解析

    使用ftpClient下載ftp上所有文件解析

    最近項(xiàng)目需要寫個(gè)小功能,需求就是實(shí)時(shí)下載ftp指定文件夾下的所有文件(包括子目錄)到本地文件夾中,保留文件到目錄路徑不變。今天小編給大家分享使用ftpClient下載ftp上所有文件的方法,需要的的朋友參考下吧
    2017-04-04
  • Java定時(shí)器問(wèn)題實(shí)例解析

    Java定時(shí)器問(wèn)題實(shí)例解析

    這篇文章主要結(jié)合實(shí)例介紹了java當(dāng)中的定時(shí)器的一些問(wèn)題,有需要的朋友可以參考一下
    2017-04-04

最新評(píng)論