Cron表达式使用问题总结

因博主最近使用定时任务时,关于Cron表达式的使用,出现问题,故记录一下问题,避免大家踩坑

1 Cron表达式简介

Cron,又称计划任务: 是任务在约定的时间执行已经计划好的工作. 在Linux中,我们经常用到 cron 服务器来完成这项工作。cron服务器可以根据配置文件约定的时间来执行特定的任务。

2 Cron表达式的格式

Cron的语法格式有如下两种:

  • Seconds Minutes Hours DayofMonth Month DayofWeek Year
  • Seconds Minutes Hours DayofMonth Month DayofWeek

1 表达式每个域支持的值

名称是否必须允许值特殊字符
0-59, - * /
0-59, - * /
0-23, - * /
1-31, - * ? / L W C
1-12 或 JAN-DEC, - * /
1-7 或 SUN-SAT, - * ? / L C #
空 或 1970-2099, - * /

ps:

1 月份和星期的名称是不区分大小写的。FRI 和 fri 是一样的。 域之间有空格分隔.

2 表中周的1-7用颜色突出,后续解释使用中出现的问题.

2 特殊字符说明

*星号

使用星号(*) 指示着你想在这个域上包含所有合法的值。例如,在月份域上使用星号意味着每个月都会触发这个 trigger。

表达式样例: 
0 * 17 * * ? 
意义:每天从下午5点到下午5:59中的每分钟激发一次trigger.

? 问号

? 号只能用在日和周域上,但是不能在这两个域上同时使用。你可以认为 ? 字符是 "我并不关心在该域上是什么值。" 这不同于星号,星号是指示着该域上的每一个值。? 是说不为该域指定值。 

表达式样例: 
0 10,44 14 ? 3 WED 
意义:在三月中的每个星期三的下午 2:10 和 下午 2:44 被触发。

, 逗号

逗号 (,) 是用来在给某个域上指定一个值列表的。例如,使用值 0,15,30,45 在秒域上意味着每15秒触发一个 trigger。 

表达式样例: 
0 0,15,30,45 * * * ? 
意义:每刻钟触发一次 trigger。

/ 斜杠

斜杠 (/) 是用于时间表的递增的。我们刚刚用了逗号来表示每15分钟的递增,但是我们也能写成这样 0/15。

表达式样例: 
0/15 0/30 * * * ? 
意义:在整点和半点时每15秒触发 trigger。

-中划线

中划线 (-) 用于指定一个范围。例如,在小时域上的 3-8 意味着 "3,4,5,6,7 和 8 点。"  域的值不允许回转,所以像 50-10 这样的值是不允许的。 

表达式样例: 
0 45 3-8 ? * * 
意义:在上午的3点至上午的8点的45分时触发 trigger。

L 字母

L 说明了某域上允许的最后一个值。它仅被日和周域支持。当用在日域上,表示的是在月域上指定的月份的最后一天。

表达式样例:
0 0 8 L * ? 
意义: 是在每个月最后一天的上午 8:00 触发 trigger。在月域上的 * 说明是 "每个月"。

W 字母 WorkDay

W 字符代表着平日 (Mon-Fri),并且仅能用于日域中。它用来指定离指定日的最近的一个平日。W 只能用在指定的日域为单天,不能是范围或列表值。

例如,日域中的 15W 意味着 "离该月15号的最近一个平日。" 假如15号是星期六,那么 trigger 会在14号(星期五)触发.

#井号

字符仅能用于周域中。它用于指定月份中的第几周的哪一天。
例如,如果你指定周域的值为 6#3,它意思是某月的第三个周五 (6=星期五,#3意味着月份中的第三周)。另一个例子 2#1 意思是某月的第一个星期一 (2=星期一,#1意味着月份中的第一周)。注意,假如你指定 #5,然而月份中没有第 5 周,那么该月不会触发。 

3 使用遇到的问题

1 关于问题

​ 博主的要求时, 每周日0点执行定时任务. 验证定时任务功能时, 博主都是具体到每天的几点钟几分钟去验证,验证结果没有问题,定时任务可以运行,且逻辑正常.

​ 结合自己的理解和网上资料,我给出的Cron表达式是 0 0 0 ? * 1 因为这个表达式,自己没有去验证,导致定时任务出现问题. 定时任务周日0点没有执行,到了周一0点才执行的,我一脸懵. 很奇怪怎么会这样.经过查看资料发现,cron表达式准备很多.

常用的quartz框架中cron表达式,:

  • 1-7 分别是,周日,周一…周六.

Spring中@scheduled注解中cron表达式:

  • 1-7 分别是周一,周二,…周日,且用0也表示周日,这个cron的标准,是linux/unix 和 github.com/robfig/cron 里遵循的.

2 解决方法

1 Cron表达式中周域使用英文表示

​ 因为Cron表达式周域中支持英文单词缩写, 即用SUN, MON, TUE, WED, THU, FRI, SAT,分别表示星期天,星期一,…星期六.

​ 采用英文缩写的方式,能直接避免上述的问题,因为不管什么Cron标准,都是支持英文缩写的,所以使用通用的表达方式,可以更好的兼容不同的场景使用该表达式.

4 总结

​ 定义的每一个Cron表达式,一定要测试验证一下. 因为整个定时任务,所有做的一切都是为了任务能够在准确的时间去执行指定的任务. 二者缺一不可.

所以我们一定要以测试结果为依据.即我们要以唯物辩证的角度客观地去认证事务的结果.千万不要觉得不重要,事情很小,没问题,而疏忽掉这个细节.

毕竟,千里之堤,溃于蚁穴;百步之室,以突隙之烟焚的故事不算少数.

  • 2
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
要在Java中解析cron表达式,你可以使用cron-utils库。首先,你需要将cron-utils添加到你的项目依赖中。根据引用的Maven配置,你可以将以下代码添加到你的pom.xml文件中: ``` <dependency> <groupId>com.cronutils</groupId> <artifactId>cron-utils</artifactId> <version>9.1.5</version> </dependency> ``` 引用中的代码展示了cron-utils库中提供的CronType枚举类,它定义了不同的cron表达式类型,包括CRON4JQUARTZ、UNIX和SPRING。 为了解析cron表达式,你可以使用CronParser类,如引用中的代码所示。首先,你需要创建一个CronParser对象,并使用CronDefinitionBuilder类的instanceDefinitionFor方法根据CronType.SPRING来定义cron表达式的语法。然后,你可以使用CronParser的parse方法将cron表达式字符串解析为Cron对象。 接下来,你可以使用ExecutionTime类的forCron方法来获取给定cron表达式的执行时间。你可以使用ZonedDateTime类来表示时间,并使用ZonedDateTime.now()来获取当前时间。然后,你可以使用ExecutionTime的nextExecution方法来获取下一次执行的时间。引用中的代码使用循环来获取指定数量的执行时间,并将它们存储在一个列表中。 最后,你可以使用DateTimeFormatter类来格式化ZonedDateTime对象为所需的日期时间格式,并将格式化后的时间存储在另一个列表中。最后,你可以返回列表作为解析cron表达式后的执行时间。 总结起来,这是解析cron表达式的基本步骤: 1. 添加cron-utils依赖到你的项目中,参考引用中的Maven配置。 2. 创建一个CronParser对象,并使用CronDefinitionBuilder类的instanceDefinitionFor方法定义cron表达式的语法。 3. 使用CronParser的parse方法将cron表达式字符串解析为Cron对象。 4. 创建一个ExecutionTime对象,并使用forCron方法获取给定cron表达式的执行时间。 5. 创建一个ZonedDateTime对象来表示时间,并使用ZonedDateTime.now()获取当前时间。 6. 使用ExecutionTime的nextExecution方法获取下一次执行的时间,并将其存储在一个列表中。 7. 使用DateTimeFormatter类来格式化ZonedDateTime对象为所需的日期时间格式,并将格式化后的时间存储在另一个列表中。 8. 返回列表作为解析cron表达式后的执行时间。 希望这个解答对你有帮助。如果你有任何其他问题,请随时问我。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值