最近栈长用 Spring Boot 写了一个定时任务:

@Scheduled(cron = \"0/10 * * * * ? *\")
public void execute() {
    ...
}

Spring Boot 实现定时任务确实很简单,其实是从 Spring 3.1 开始,定时任务的编写就变得非常简单,只需要几个注解就能快速开启计划任务的支持。

可是当我把代码写完,启动就报错:

Caused by: java.lang.IllegalStateException: Encountered invalid @Scheduled method 'xxx': Cron expression must consist of 6 fields (found 7 in \"0/10 * * * * ? *\")
    at org.springframework.scheduling.annotation.ScheduledAnnotationBeanPostProcessor.processScheduled(ScheduledAnnotationBeanPostProcessor.java:511) ~[spring-context-5.3.1.jar:5.3.1]
    at org.springframework.scheduling.annotation.ScheduledAnnotationBeanPostProcessor.lambda$null$1(ScheduledAnnotationBeanPostProcessor.java:374) ~[spring-context-5.3.1.jar:5.3.1]
    at java.lang.Iterable.forEach(Iterable.java:75) ~[na:1.8.0_261]
    at org.springframework.scheduling.annotation.ScheduledAnnotationBeanPostProcessor.lambda$postProcessAfterInitialization$2(ScheduledAnnotationBeanPostProcessor.java:374) ~[spring-context-5.3.1.jar:5.3.1]
    at java.util.LinkedHashMap.forEach(LinkedHashMap.java:684) ~[na:1.8.0_261]
    at org.springframework.scheduling.annotation.ScheduledAnnotationBeanPostProcessor.postProcessAfterInitialization(ScheduledAnnotationBeanPostProcessor.java:373) ~[spring-context-5.3.1.jar:5.3.1]
    at org.springframework.beans.factory.support.AbstractAutowireCapableBeanFactory.applyBeanPostProcessorsAfterInitialization(AbstractAutowireCapableBeanFactory.java:444) ~[spring-beans-5.3.1.jar:5.3.1]
    at org.springframework.beans.factory.support.AbstractAutowireCapableBeanFactory.initializeBean(AbstractAutowireCapableBeanFactory.java:1792) ~[spring-beans-5.3.1.jar:5.3.1]
    at org.springframework.beans.factory.support.AbstractAutowireCapableBeanFactory.doCreateBean(AbstractAutowireCapableBeanFactory.java:609) ~[spring-beans-5.3.1.jar:5.3.1]
    ... 16 common frames omitted

什么鬼?

错误描述大概是说我的 Cron 表达式有问题,Cron 表达式只能由 6 位组成,但我的表达式有 7 位,不能啊,表达式也确实没问题的啊。。。

点进去看下 @Scheduled 注解的 cron 参数源码:

确实只有 6 位,并没有表示年份的占位。

继续看源码,CronSequenceGenerator 在 Spring 5.3 中已经标识废弃了,需要转到:

org.springframework.scheduling.support.CronExpression

再点进去看 CronExpression 类的源码:

6 个字段的意思很清楚了,具体可以参考:

示例表达式:

\"0 0 * * * *\" = the top of every hour of every day.
\"*/10 * * * * *\" = every ten seconds.
\"0 0 8-10 * * *\" = 8, 9 and 10 o'clock of every day.
\"0 0 6,19 * * *\" = 6:00 AM and 7:00 PM every day.
\"0 0/30 8-10 * * *\" = 8:00, 8:30, 9:00, 9:30, 10:00 and 10:30 every day.
\"0 0 9-17 * * MON-FRI\" = on the hour nine-to-five weekdays
\"0 0 0 25 12 ?\" = every Christmas Day at midnight
\"0 0 0 L * *\" = last day of the month at midnight
\"0 0 0 L-3 * *\" = third-to-last day of the month at midnight
\"0 0 0 1W * *\" = first weekday of the month at midnight
\"0 0 0 LW * *\" = last weekday of the month at midnight
\"0 0 0 * * 5L\" = last Friday of the month at midnight
\"0 0 0 * * THUL\" = last Thursday of the month at midnight
\"0 0 0 ? * 5#2\" = the second Friday in the month at midnight
\"0 0 0 ? * MON#1\" = the first Monday in the month at midnight

并且支持以下代替写法:

\"@yearly\" (or \"@annually\") to run un once a year, i.e. \"0 0 0 1 1 *\",
\"@monthly\" to run once a month, i.e. \"0 0 0 1 * *\",
\"@weekly\" to run once a week, i.e. \"0 0 0 * * 0\",
\"@daily\" (or \"@midnight\") to run once a day, i.e. \"0 0 0 * * *\",
\"@hourly\" to run once an hour, i.e. \"0 0 * * * *\".

如:每年:0 0 0 1 1 *,就可以写成:@yearly,除了每年,还有每月、每月、每周、每天、每小时,可惜……没有每分、每秒!

Spring Task 我们可以将它看成是一个轻量级的 Quartz,而且使用起来比 Quartz 要简单许多,使用了类似于 cron 的表达式,并扩展了平常的 UN*X 定义,看来和平常的 cron 表达式还不太一样,位数不同,也存在兼容问题,真坑。。。

既然知道了问题所在java技术栈,再回到之前错误,那就只能把最后的表示 “年” 的字段去掉,6 位就够了,第7 位不要,默认就是表示每年。

那么问题来了,我想在具体的某年,如 2025/02/25 这天执行任务,咋整?

没办法,Spring Task 不能指定年份,那就只能换成 Quartz,Quartz 也更强大,所以,如果是简单的定时任务,Spring Task 就搞定了java技术栈,复杂的建议还是使用 Quartz。

另外,注意,Spring Task 的 cron 表达式并不是完全兼容通常的 cron 表达式,最好根据以上官方的文档来编写,以免到了线上不执行那就完蛋了。

本文所有示例代码已上传:javastacks/spring-boot-best-practice

最后,觉得我的文章对你用收获的话,动动小手,给个在看、转发,原创不易,栈长需要你的鼓励。

版权申明:本文系 “Java技术栈” 原创,原创实属不易,转载、引用本文内容请注明出处,禁止抄袭、洗稿,请自重,尊重他人劳动成果和知识产权。

发表回复

后才能评论

本站所有资源版权均属于原作者所有,这里所提供资源均只能用于参考学习用,请勿直接商用。若由于商用引起版权纠纷,一切责任均由使用者承担。更多说明请参考 VIP介绍。

最常见的情况是下载不完整: 可对比下载完压缩包的与网盘上的容量,若小于网盘提示的容量则是这个原因。这是浏览器下载的bug,建议用百度网盘软件或迅雷下载。 若排除这种情况,可在对应资源底部留言,或联络我们。

对于会员专享、整站源码、程序插件、网站模板、网页模版等类型的素材,文章内用于介绍的图片通常并不包含在对应可供下载素材包内。这些相关商业图片需另外购买,且本站不负责(也没有办法)找到出处。 同样地一些字体文件也是这种情况,但部分素材会在素材包内有一份字体下载链接清单。

如果您已经成功付款但是网站没有弹出成功提示,请联系站长提供付款信息为您处理

源码素材属于虚拟商品,具有可复制性,可传播性,一旦授予,不接受任何形式的退款、换货要求。请您在购买获取之前确认好 是您所需要的资源