天天看點

spring定時任務cron為每季度第一個工作日轉換為cron表達式

最近系統有個需求,執行一個定時任務發送待辦事項,一年有六次,分别為每季度第一個工作日、第二季度和第四季度結束後的第十個工作日,可頭疼壞了

還好我們系統有個工作日表,于是思路就是先查詢出每季度的第一天,然後在表裡查出大于此日期的第一個工作日,然後利用cronUtil轉換成cron表達式,開啟定時任務就ok了。

下面附上查詢季度第一天的sql

/* 本季度的第一天 */
concat(year(curdate()),'-',elt(quarter(curdate()),1,4,7,10),'-',1)
/* 下個季度的第一天 */
date_add(LAST_DAY(MAKEDATE(EXTRACT(YEAR FROM curdate()),1) + interval QUARTER(curdate())*3-1 month),interval 1 day )
/* 本半年度的第一天   我的需求是這樣,你們可以自由變動*/
concat(year(curdate()),'-07-01')
/* 明年半年度的第一天 */
concat(year(date_add(curdate(),interval 1 year )),'-07-01')
/* 查詢本季度第一個工作日    需要先有個工作日表哈*/
select 日期
        from 工作日表
        where 日期>=concat(year(curdate()),'-',elt(quarter(curdate()),1,4,7,10),'-',1)  and 是否為工作日=1 limit 1
           

查詢出來的日期通過cronUtil轉換成cron表達式,下面附上cronUtil

package com.ncamc.sfa.schedule.util;

import java.text.SimpleDateFormat;
import java.util.Date;

/***
 *  功能描述:日期轉換cron表達式
 */
public class CronUtil {
    /***
     *  功能描述:日期轉換cron表達式
     * @param date
     * @param dateFormat
     * @return
     */
    public static String formatDateByPattern(Date date, String dateFormat){
        SimpleDateFormat sdf = new SimpleDateFormat(dateFormat);
        String formatTimeStr = null;
        if (date != null) {
            formatTimeStr = sdf.format(date);
        }
        return formatTimeStr;
    }
    /***
     * @param date  : 時間點
     * @return
     */
    public static String getCron(Date  date){
        String dateFormat="ss mm HH dd MM ?";
        return formatDateByPattern(date, dateFormat);
    }
}
           

注意:spring的scheduled的cron不能有年份,如果需要年份,需要合并quartz來實作了

由于我的需求是每個季度的第一個工作日,有可能今年跟下一年的季度第一個工作日是不一樣的,是以定時任務也不能寫死,這就需要在每次定時任務執行的時候,要修改此任務的cron,也就是擷取相應任務的線程id,先停止任務,然後放入新的cron重新啟動任務,可以參考我寫的上一篇文章。

可參考這個:

https://blog.csdn.net/success321/article/details/116985433