Spring 实现动态定时任务

一、前言

平时我们使用 Spring 去实现一个单体项目的定时任务,用得比较多的是 @Scheduled 注解,但是该注解的局限性是不能动态的更改定时任务的时间;这时就需要使用其他方式去实现动态的定时任务了,Spring 给我们提供了 SchedulingConfigurer 接口,使用它就能够实现动态的定时任务。

二、使用

直接实现 SchedulingConfigurer 接口即能做到动态的修改定时任务的时间,下面先写一个配置类 TestScheduleConfig。注:别忘了添加 @EnableScheduling 注解

/**
 * 动态定时任务测试
 *
 * @author zxb
 * @date 2022-06-08 17:53
 **/
@Configuration
@EnableScheduling
@Log4j2
public class TestScheduleConfig implements SchedulingConfigurer {
    /**
     * cron 表达式,可以设置;默认 1s 执行一次
     */
    @Getter
    @Setter
    private String cron = "0/1 * * * * ? ";

    private final Runnable runnable = () -> {
        if (log.isInfoEnabled()) {
            log.info("运行的任务......");
        }
    };

    @Override
    public void configureTasks(ScheduledTaskRegistrar taskRegistrar) {
        taskRegistrar.addTriggerTask(runnable, triggerContext -> {
            // 下一次执行定时任务的 cron 表达式
            CronTrigger trigger = new CronTrigger(cron);
            if (log.isInfoEnabled()) {
                log.info("执行的 cron 为:".concat(cron));
            }
            return trigger.nextExecutionTime(triggerContext);
        });
    }
}

需要重写 configureTasks 方法,该方法会在应用启动完成后自动调用,其中参数为 ScheduledTaskRegistrar 类型,从下图看来添加任务有几种类型,分别是固定 corn 表达式的任务(与@Scheduled 一样)、固定延时与固定速率任务、触发式任务。本次使用的是 addTriggerTask 方法。

在这里插入图片描述

三、测试

我们默认给定时任务设置每秒执行一次任务,然后过 10 秒修改为 10 秒执行一次,过 20 秒修改为每 2 秒执行一次,下面看看效果。

/**
 * @author zxb
 * @date 2022-06-08 17:58
 **/
@SpringBootTest
@Log4j2
public class ScheduleTest {
    @Autowired
    private TestScheduleConfig testScheduleConfig;

    @Test
    @SneakyThrows
    public void test() {
        TimeUnit.SECONDS.sleep(10);
        log.info("更改 cron 时间为 10 秒一次");
        testScheduleConfig.setCron("0/10 * * * * ? ");

        TimeUnit.SECONDS.sleep(20);
        log.info("更改 cron 时间为 2 秒一次");
        testScheduleConfig.setCron("0/2 * * * * ? ");
        
        TimeUnit.SECONDS.sleep(60);
    }
}

执行结果如下图所示,控制台输出更改 cron 表达式后,任务的执行时间并没有立即更改,而是更改下一次任务的执行时间。

在这里插入图片描述

四、总结

我们在需要动态设置定时任务而 @Scheduled 注解不满足应用场景的时,可以使用 SchedulingConfigurer 接口进行代替,使用起来也很简单。但该方案仅仅适用于单体服务,当分布式时请使用类似 XXL-JOB 这样的支持分布式的定时任务框架,或者加个分布式锁(使用数据库实现乐观锁也可以)防止多节点同时执行任务,导致任务重复执行问题。

  • 3
    点赞
  • 16
    收藏
    觉得还不错? 一键收藏
  • 1
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值