根据cron表达式获取最近几次执行的时间

根据cron表达式获取最近几次执行的时间

最近有个新需求,就是现在有个定时任务,前端需要展示出最近一次的具体执行时间:

具体可有以下两种做法(可能更多),个人推荐方式一
  • 方式一:指定获取的最近执行的次数

首先maven引入依赖(本来就有定时任务的,此步骤仅又来做个人测试)

<dependency>
      <groupId>org.quartz-scheduler</groupId>
      <artifactId>quartz</artifactId>
      <version>2.2.1</version>
 </dependency>

直接贴上method

/**
 *
 * @param cronExpression    cron表达式
 * @param numTimes    下一(几)次运行的时间
 * @return
 */
public static List<String> getNextExecTime(String cronExpression,Integer numTimes) {
        List<String> list = new ArrayList<>();
        CronTriggerImpl cronTriggerImpl = new CronTriggerImpl();
        try {
            cronTriggerImpl.setCronExpression(cronExpression);
        } catch(ParseException e) {
            e.printStackTrace();
        }
        // 这个是重点,一行代码搞定
        List<Date> dates = TriggerUtils.computeFireTimes(cronTriggerImpl, null, numTimes);
        SimpleDateFormat dateFormat = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");
        for (Date date : dates) {
            list.add(dateFormat.format(date));
        }
        return list;
    }
  • 方式二:获取指定时间内(可以自己指定年或月或日)所有的执行时间,然后在所有的时间内取前几个(不推荐,当触发时间过短,程序响应时间非常长)

废话不多说,直接上method

   /**
     * @param cronExpression cron表达式
     * @param numTimes       下一(几)次运行的时间
     * @return
     */
    public static List<String> getRecentExecTime(String cronExpression, Integer numTimes) {
        List<String> list = new ArrayList<>();
        try {
            CronTriggerImpl cronTrigger = new CronTriggerImpl();
            cronTrigger.setCronExpression(cronExpression);
            // 这里写要准备猜测的cron表达式
            Calendar calendar = Calendar.getInstance();
            Date now = calendar.getTime();
            // 把统计的区间段设置为从现在到2年后的今天(主要是为了方法通用考虑,如那些1个月跑一次的任务,如果时间段设置的较短就不足20条)
            calendar.add(Calendar.YEAR, 2);
            // 这个是重点,一行代码搞定
            List<Date> dates = TriggerUtils.computeFireTimesBetween(cronTrigger, null, now, calendar.getTime());
            SimpleDateFormat dateFormat = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");
            for(int i = 0; i < dates.size(); i++) {
                // 这个是提示的日期个数
                if(i < numTimes) {
                    list.add(dateFormat.format(dates.get(i)));
                }else {
                    break;
                }
            }

        } catch(ParseException e) {
            e.printStackTrace();
        }
        return list;
    }
### 使用 Quartz 计算 Cron 表达式的上一次执行时间 为了计算基于 Cron 表达式的前一次任务执行时间,可以利用 `org.quartz.CronExpression` 类中的方法。此方法涉及创建一个表示过去某个时刻的日期对象,并使用该类提供的功能向前寻找最接近的有效触发时间。 对于获取上一次执行时间的操作,可以通过调整当前时间向过去偏移一定量的时间单位(如几秒钟),从而使得调用 `getNextValidTimeAfter()` 方法返回的结果实际上是相对于现在而言过去的那个最近有效时间点[^1]。 下面是一个 Java 实现的例子: ```java import org.quartz.CronExpression; import java.text.ParseException; public class PreviousExecutionTimeFinder { /** * 获取指定 cron 表达式对应的上一次执行时间. * * @param cronExpression cron表达式字符串形式 * @return 上一次执行的具体毫秒数, 如果无法解析则返回0L */ public static long getCronPreviousTimeByTempInfo(String cronExpression){ try { if (!CronExpression.isValidExpression(cronExpression)) { return 0L; } // 创建一个新的 CronExpression 对象并初始化它 CronExpression cron = new CronExpression(cronExpression); // 构建一个稍微早于现在的日期作为起点 Date slightlyBeforeNow = new Date(System.currentTimeMillis() - 5000); // 减去五秒 // 调整时区以匹配服务器默认设置 cron.setTimeZone(TimeZone.getDefault()); // 寻找在此刻之前的最后一个有效的触发时间 Date previousValidTime = null; synchronized (cron) { previousValidTime = cron.getPrevFireTime(); // 若未找到,则尝试通过不断回溯直到找到为止 while (previousValidTime == null || previousValidTime.after(slightlyBeforeNow)){ slightlyBeforeNow.setTime(previousValidTime != null ? previousValidTime.getTime()-1 : System.currentTimeMillis()-10000); previousValidTime = cron.getPrevFireTime(); } } return previousValidTime!=null?previousValidTime.getTime():0L; } catch (ParseException | NullPointerException e) { e.printStackTrace(); } return 0L; } } ``` 需要注意的是,在实际应用中可能还需要考虑更多边界情况处理以及性能优化等问题。上述代码片段展示了基本逻辑实现方式之一[^3]。
评论 7
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值