人生有涯,学海无涯
Spring 的定时任务想必大家多多少少都用过,经过 Spring 团队的封装,大家使用起来非常的方便和简洁,那关于 定时任务的真正使用还有哪些你不知道的事呢?下面我们一起来看一下吧。
官方条件
在使用Scheduled 注解前,我们先看下,官方给出了使用条件
An annotation that marks a method to be scheduled. Exactly one of the cron(), fixedDelay(), or fixedRate() attributes must be specified.
The annotated method must expect no arguments. It will typically have a void return type; if not, the returned value will be ignored when called through the scheduler.
Processing of
@Scheduledannotations is performed by registering aScheduledAnnotationBeanPostProcessor. This can be done manually or, more conveniently, through the<task:annotation-driven/>element or @EnableSchedulingannotation.
意思是说,在使用 Scheduled 注解的时候,必须设置 cron(),fixedDelay(), fixedRate() 三个中的一个属性。该注解使用的方法不期望接收任何参数,并且要使用 void 返回类型,即使设置了返回值,也会在回调的时候被忽略。第三句是说,我们在使用 Scheduled 注解的时候需要注册 ScheduledAnnotationBeanPostProcessor ,可以通过手动,或者配置 <task:annotation-driven/> 或者在 Springboot 项目中,直接在启动类上添加 @EnableScheduling 注解都可以。总结一下就是
设置定时策略;
不设置返回类型;
注册
processor。
使用案例
在平时使用的时候我们基本上只要加一个注解,然后再配置上相应的 cron 参数就可以了。如下所示:

Scheduled 参数配置说明
cron()配置cron是一个六位或者七位的字符串,每一位都对应的相应的含义,如上0 0 0 * * ?表示的是每天 0 点执行。表达式从左往右依次代表的含义是Seconds Minutes Hours DayofMonth Month DayofWeek Year或Seconds Minutes Hours DayofMonth Month DayofWeek详细的配置方式,读者感兴趣可以自行研究,而且网上也有根据条件自动生产表达式的,可以参考。fixedDelay()方式,执行完毕后调用官方解释
Execute the annotated method with a fixed period in milliseconds between the end of the last invocation and the start of the next.意思是说在上一次执行完毕和下一次开始调用的中间,延迟一段时间,以毫秒为单位。
fixedRate()方式,开始执行后延迟调用官方解释
Execute the annotated method with a fixed period in milliseconds between invocations.固定一个时间间隔进行前后方法的调用。
这几种的配置方式还是比较简单的,可以根据自己的场景以及需求来设置。可以说都还是比较常用的。
注意
这里结合自身项目,跟大家介绍几个在使用的过程中容易遇到的一些问题。
单线程问题
Spring 的任务调用默认使用的是单线程模式,这个如果大家没有在生产上真正遇到过可能不会发现有问题,毕竟本地和测试环境数据少,不容易出现,但是在真正复杂的生产环境中,还是要十分注意的。

由于 Spring 的默认实现是单线程,所以在生产的时候如果任务一旦多了,而且耗时长了,就会出现很多意想不到的情况,比如会发现某个任务不执行,或者没有按照指定的时间运行。这种情况往往是很恐怖的,如果说对实时性要求不高的话,影响不大,如果实时性要求很高的话,那绝对不能容忍。
这个时候我们需要自定义taskScheduler ,通过如下方式
import org.springframework.beans.factory.annotation.Value;
import org.springframework.context.annotation.Configuration;
import org.springframework.scheduling.annotation.SchedulingConfigurer;
import org.springframework.scheduling.config.ScheduledTaskRegistrar;
/**
* <br>
* <b>功能:</b><br>
* <b>作者:</b>@author 子悠<br>
* <b>日期:</b>2019-08-22 17:51<br>
* <b>详细说明:</b>无<br>
*/
@Configuration
public class ScheduledThreadPoolConfig implements SchedulingConfigurer {
@Value("${thread.pool.corePoolSize:10}")
private int corePoolSize;
@Override
public void configureTasks(ScheduledTaskRegistrar scheduledTaskRegistrar) {
scheduledTaskRegistrar.setScheduler(Executors.newScheduledThreadPool(corePoolSize));
}
}
通过实现 SchedulingConfigurer 接口中的 configureTasks 方法,来创建一个 ScheduledTask 并注入到 ScheduledTaskRegistrar 中。
这样就可以自定义线程数,从而避免上述问题的存在。
多实例问题
另外还有一个问题就是如果一个模块定时任务很多,最好可以封装为一个独立的任务调度模块,对于负责场景的任务调度可以采用一些开源的调度框架,比如 XXL-JOB,Azkaban。
如果只是一些简单的定时任务,不需要引入开源框架的时候,那么自己开发后在部署的时候需要注意多实例会重复运行的情况,意思是说一个定时任务模块,如果部署了两个实例就会导致任务重复执行。这种时候只能采用单实例部署,单实例部署的时候需要加守护进程和程序监控,避免程序出问题。
如果是在需要高可用,可以自己开发调度模块或者采用开源的框架实现。大部分场景 XXL-JOB 应该都可以满足。
总结
今天给大家介绍了 Spring 框架中的 Scheduled 组件的简单使用,以及在使用过程中注意的一些事项,希望对大家有帮助。
人生有涯,学海无涯,欢迎大家到我们《Java 极客技术》的知识星球中一起进步,一起成长。
年轻人我看你骨骼清秀,是练武奇才
前1000人,50元/年
还有81位优惠名额
长按二维码▽

点转发,看看你朋友能属于3%吗
本文深入探讨Spring框架中Scheduled组件的使用方法,包括cron表达式、fixedDelay和fixedRate的配置,以及在生产环境中可能遇到的单线程问题和多实例问题的解决方案。
733

被折叠的 条评论
为什么被折叠?



