@Scheduled不生效了:有一天我的定时任务突然不执行了

一、问题:

之前定时任务一致好好的,某天突然服务没有执行,重启之后也不执行,本地起服务也不执行。。。。

二、检查

@EnableScheduling 这个注解一直都有加的

声明为 

@Scheduled(fixedDelay = 2) 的任务可以很好的执行
@Scheduled(cron = "0 30 19 * * ?") cron表达式的 到点了就是不执行

三、解决

上网看了, 可能原因 任务是懒加载的,调用一次之后才会加载执行(但是为什么 之前好好的呢?)

解决方法:手动配置了 定时任务的  ScheduledThreadPoolExecutor  代码如下:

@Configuration
public class ScheduleConfig implements SchedulingConfigurer {

    @Override
    public void configureTasks(ScheduledTaskRegistrar scheduledTaskRegistrar) {
        scheduledTaskRegistrar.setScheduler(new ScheduledThreadPoolExecutor(Runtime.getRuntime().availableProcessors(),
                new ThreadFactory() {
                    @Override
                    public Thread newThread(Runnable r) {
                        return new Thread(r,"my-schedule");
                    }
                }));
    }
}

然后好了

四、附送

基于 @Scheduled 的定时任务,其实会在 bean 实例化阶段 的  BeanPostProcessor(的具体子类实现ScheduledAnnotationBeanPostProcessor  的 postProcessAfterInitialization)将 所有附带  @Scheduled注解的方法检测出,分析对应的 参数内容, 然后加入各个任务队列之中。

我们配置了 定时任务 使用自己的 ScheduledThreadPoolExecutor  内部其实 基于 DelayQueue,每次任务执行完成之后会计算是否需要下次执行,以及下次执行的时间,然后将任务在放入队列之中。

关于 @EnableScheduling,不加这个注解,在项目启动时  @Scheduled(cron = "0 30 19 * * ?") 这个不会执行, 但是 @Scheduled(fixedDelay = 2)会执行, 因为  initialDelay是默认值的缘故,在将任务加入队列之前会 先 调用一下当前的任务,所以项目启动时 会执行一次。

猜你喜欢

转载自blog.csdn.net/weixin_37882382/article/details/82217729