一尘不染

Quartz:永远不会执行的Cron表达式

spring

我使用Spring应用程序上下文来处理Java Web应用程序。在这种情况下,我使用Quartz定义了计划作业。这些作业由.properties文件中定义的cron触发。

Spring上下文嵌入战争中,而.properties文件位于应用程序服务器(在这种情况下为Tomcat)上。

这很好,并允许根据环境(开发,集成,生产等)定义不同的克朗。

现在,在我自己的计算机上本地运行此应用程序时,我不希望执行这些作业。有没有办法编写永远不会触发的cron表达式?


阅读 3359

收藏
2020-04-16

共1个答案

一尘不染

TL; DR
在Quartz 1中,你可以使用以下cron :(59 59 23 31 12 ? 2099最后一个有效日期)。
在Quartz 2中,你可以使用以下cron:0 0 0 1 1 ? 2200

将来使用表达式

使用进行了一些快速测试org.quartz.CronExpression。

String exp = "0 0 0 1 1 ? 3000";
boolean valid = CronExpression.isValidExpression(exp);
System.out.println(valid);
if (valid) {
    CronExpression cronExpression = new CronExpression(exp);
    System.out.println(cronExpression.getNextValidTimeAfter(new Date()));
}

当我这样做时String exp = "# 0 0 0 1 1 ?";,isValid测试返回false。

有了上面给出的示例,输出如下:

true
null

含义:

  • 该表达式有效;
  • 没有与该表达式匹配的即将到来的日期。

但是,要使调度程序接受cron触发器,后者必须与将来的日期匹配。

我尝试了几年,发现一旦年份超过2300,Quartz似乎就不再打扰了(尽管我在Quartz 2的文档中没有提到该年份的最大值)。可能会有更清洁的方法来执行此操作,但这将满足我的需求。

因此,最后,我建议的时间表是0 0 0 1 1 ? 2200。

Quartz 1 variant

请注意,在Quartz 1中,2099是最后一个有效年份。因此,你可以调整cron表达式以使用Maciej Matys的建议:59 59 23 31 12 ? 2099

替代方法:使用过去的日期
Arnaud Denoyelle提出了一个更优雅的建议,上面的测试证明了这一说法是正确的:与其选择一个很远的日期,不如选择一个很远的日期:

0 0 0 1 1 ? 1970 (根据Quartz文档的第一个有效表达式)。

但是,此解决方案不起作用。

hippofluff强调说Quartz过去将检测到一个表达式,将不再执行,因此将引发异常。

org.quartz.SchedulerException: Based on configured schedule, the given trigger will never fire.

这似乎已经在Quartz 很久了。

获得的经验:测试并非万无一失
这凸显了我的测试的弱点:如果要测试a CronExpression,请记住它必须具有nextValidTime1。否则,你将其传递给的调度程序将简单地拒绝它,并带有上述异常。

我建议修改测试代码如下:

String exp = "0 0 0 1 1 ? 3000";
boolean valid = CronExpression.isValidExpression(exp);
if (valid) {
    CronExpression cronExpression = new CronExpression(exp);
    valid = cronExpression.getNextValidTimeAfter(new Date()) != null;
}
System.out.println("Can I use <" + exp + ">? " + (valid ? "Go ahead!" : "This shall fail."));

到此为止:无需思考,只需阅读输出即可。

2020-04-16