1. 程式人生 > >spring boot專案中處理Schedule定時任務

spring boot專案中處理Schedule定時任務

預設,springboot已經支援了定時任務Schedule模組,所以一般情況已經完全能夠滿足我們的實際需求,一般來說,沒有必要在加入其他類似於:quartz

另外,在這裡提一個實際專案中,關於定時任務的架構上的一些考慮:

一般來說,實際專案中,為了提高服務的響應能力,我們一般會通過負載均衡的方式,或者反向代理多個節點的方式來進行。通俗點來說,我們一般會將專案部署多例項,或者說部署多份,每個例項不同的啟動埠。但是每個例項的程式碼其實都是一樣的。如果我們將定時任務寫在我們的專案中,就會面臨一個麻煩,就是比如我們部署了3個例項,三個例項一啟動,就會把定時任務都啟動,那麼在同一個時間點,定時任務會一起執行,也就是會執行3次,這樣很可能會導致我們的業務出現錯誤。

一般來說,我們有幾種簡單的辦法來處理:

1、配置檔案中增加自定義配置,通過開關來進行控制:比如增加:schedule=enable , schedule=disable,這樣在我們的實際程式碼中,在進行判斷,也就是我們可以通過配置,達到,只有一個例項真正執行定時任務,其他的是例項不執行。但是,這種做法實際是還是定時任務都啟動,只是在執行中,我們人工來進行判斷,執行於不執行真正的處理邏輯。 2、邏輯分離,就是我們將真正要定時任務處理的邏輯,寫成rest服務,或者rpc服務,然後我們可以新建一個單獨的定時任務專案,這個專案應該是沒有任何的業務程式碼的,他純粹只有定時任務功能,幾點啟動,或者每隔多少時間啟動,啟動後,通過rest或者rpc的方式,呼叫真正處理邏輯的服務。同時,我們甚至可以不用新建一個專案,我們通過linux的cron就可以進行。同時,這種方式還有一個好處,比如有些時候,我們的定時任務也會因為某些原因出現問題,沒有執行,那麼我們就可以通過curl 或者wget等等很多方式,再次定時任務的執行。

所以,個人一般偏向使用第二種方式,達到定時任務和業務處理的分離。

而在spring boot中,如何使用定時任務,相對比較簡單。按第二種方式,實際上,我需要新建一個專案來完成定時任務的功能,其實,我們完全可以新建一個普通的java專案,引入quartz來達到,但是這裡,我是通過spring boot來完成,新建一個spring boot專案,專案的初始化可以使用:http://start.spring.io

初始化之後,我們在spring boot的入口類Application.java中,允許支援schedule

@SpringBootApplication
@EnableScheduling
public
class Application { public static void main(String[] args) { SpringApplication.run(Application.class, args); } }

然後,新建一個執行類Jobs.java

@Component
public class Jobs {
    public final static long ONE_Minute =  60 * 1000;

    @Scheduled(fixedDelay=ONE_Minute)
    public void fixedDelayJob(){
        System.out.println(Dates.format_yyyyMMddHHmmss(new Date())+" >>fixedDelay執行....");
    }

    @Scheduled(fixedRate=ONE_Minute)
    public void fixedRateJob(){
        System.out.println(Dates.format_yyyyMMddHHmmss(new Date())+" >>fixedRate執行....");
    }

    @Scheduled(cron="0 15 3 * * ?")
    public void cronJob(){
        System.out.println(Dates.format_yyyyMMddHHmmss(new Date())+" >>cron執行....");
    }
}

這是最簡單的2種方式,多少分鐘執行一次,fixedDelay和fixedRate,單位是毫秒,所以1分鐘就是60秒×1000
他們的區別在於,fixedRate就是每多次分鐘一次,不論你業務執行花費了多少時間。我都是1分鐘執行1次,而fixedDelay是當任務執行完畢後1分鐘在執行。所以根據實際業務不同,我們會選擇不同的方式。

而還有一類定時任務,比如是每天的3點15分執行,那麼我們就需要用另外一種方式:cron表示式

cron表示式,有專門的語法,而且感覺有點繞人,不過簡單來說,大家記住一些常用的用法即可,特殊的語法可以單獨去查。
cron一共有7位,但是最後一位是年,可以留空,所以我們可以寫6位:

* 第一位,表示秒,取值0-59
* 第二位,表示分,取值0-59
* 第三位,表示小時,取值0-23
* 第四位,日期天/日,取值1-31
* 第五位,日期月份,取值1-12
* 第六位,星期,取值1-7,星期一,星期二...,注:不是第1周,第二週的意思
          另外:1表示星期天,2表示星期一。
* 第7為,年份,可以留空,取值1970-2099

cron中,還有一些特殊的符號,含義如下:

(*)星號:可以理解為每的意思,每秒,每分,每天,每月,每年...
(?)問號:問號只能出現在日期和星期這兩個位置,表示這個位置的值不確定,每天3點執行,所以第六位星期的位置,我們是不需要關注的,就是不確定的值。同時:日期和星期是兩個相互排斥的元素,通過問號來表明不指定值。比如,1月10日,比如是星期1,如果在星期的位置是另指定星期二,就前後衝突矛盾了。
(-)減號:表達一個範圍,如在小時欄位中使用“10-12”,則表示從10到12點,即10,11,12
(,)逗號:表達一個列表值,如在星期欄位中使用“1,2,4”,則表示星期一,星期二,星期四
(/)斜槓:如:x/y,x是開始值,y是步長,比如在第一位(秒) 0/15就是,從0秒開始,每15秒,最後就是0,15,30,45,60    另:*/y,等同於0/y

下面列舉幾個例子供大家來驗證:

0 0 3 * * ?     每天3點執行
0 5 3 * * ?     每天3點5分執行
0 5 3 ? * *     每天3點5分執行,與上面作用相同
0 5/10 3 * * ?  每天3點的 5分,15分,25分,35分,45分,55分這幾個時間點執行
0 10 3 ? * 1    每週星期天,3點10分 執行,注:1表示星期天    
0 10 3 ? * 1#3  每個月的第三個星期,星期天 執行,#號只能出現在星期的位置