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

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

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

@Scheduled 的詳細解析

參數(shù)詳解

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

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

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

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

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

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

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

Cron表達式、fixedRate、fixedDelay、initialDelay如何選擇     

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

示例代碼

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

@Component
public class ScheduledTasks {

    // 每天中午12點執(zhí)行(上海時區(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 的詳細解析

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

示例代碼

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

@Component
public class MultipleScheduledTasks {

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

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

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

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

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

自定義 TaskScheduler

對于更復(fù)雜的需求,比如調(diào)整線程池大小或者設(shè)置線程名稱前綴等,你可以通過自定義 TaskScheduler 來進行配置。Spring提供了幾種內(nèi)置的調(diào)度器實現(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;
    }
}

錯誤處理

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

自定義錯誤處理邏輯

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

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());
            }
        };
    }
}

最佳實踐

  • 避免長時間運行的任務(wù):盡量不要讓預(yù)定任務(wù)執(zhí)行過長的時間,因為它們可能會阻塞其他任務(wù)的執(zhí)行。如果任務(wù)有可能運行很長時間,請考慮將其拆分為更小的部分,或者使用異步處理。

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

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

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

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

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

  • 測試:編寫單元測試和集成測試來驗證定時任務(wù)的行為是否符合預(yù)期??梢允褂肕ockito或其他測試框架模擬依賴服務(wù)。

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

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

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

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

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

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

  • 長時間運行的任務(wù):盡量不要讓預(yù)定任務(wù)執(zhí)行過長的時間,因為它們可能會阻塞其他任務(wù)的執(zhí)行。如果任務(wù)有可能運行很長時間,請考慮將其拆分為更小的部分,或者使用異步處理,如通過消息隊列(MQ)來分發(fā)任務(wù)。

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

案例分析

假設(shè)你正在開發(fā)一個電子商務(wù)平臺,需要每天凌晨2點生成前一天的銷售報告。你可以使用 @Scheduled 注解來安排這個任務(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í)行生成銷售報告的邏輯
        System.out.println("Generating daily sales report at Asia/Shanghai timezone.");
    }
}

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

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

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

相關(guān)文章

  • SpringBoot使用自動配置xxxAutoConfiguration

    SpringBoot使用自動配置xxxAutoConfiguration

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

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

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

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

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

    SpringBoot整合Junit實例過程解析

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

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

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

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

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

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

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

    httpclient staleConnectionCheckEnabled獲取連接流程解析

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

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

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

    Java定時器問題實例解析

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

最新評論