如何在SpringBoot中优雅的实现定时任务

2021-10-09  本文已影响0人  Java架构师必看

大家好,我是架构君,一个会写代码吟诗的架构师。今天说一说在SpringBoot中优雅的实现定时任务,希望能够帮助大家进步!!! https://javajgs.com/archives/27921
在日常的项目开发中,往往会涉及到一些需要做到定时执行的代码,例如自动将超过24小时的未付款的单改为取消状态,自动将超过14天客户未签收的订单改为已签收状态等等,那么为了在Spring Boot中实现此类需求,我们要怎么做呢?
Spring Boot早已考虑到了这类情况,先来看看要怎么做。第一种方式是比较简单的,先搭建好Spring Boot微服务,加上这个注解 <strong>@EnableScheduling </strong>:
<div>
<pre><code>
/**

}</code></pre></div>
<p>然后编写定时任务类:</p>
<div><pre><code>/**

}</code></pre></div>
<p>@Scheduled(cron ="*/15 * * * * ?")注解表明这是一个需要定时执行的方法,里面的cron属性接收的是一个cron表达式,这里我给的是 */15 * * * * ? ,这个的意思是每隔15秒执行一次方法,对cron表达式不熟悉的同学可以百度一下用法。项目跑起来后可以看到方法被定时执行了:</p>

<p>这种方式有个缺点,那就是执行周期写死在代码里了,没有办法动态改变,要想改变只能修改代码在重新部署启动微服务。其实Spring也考虑到了这个,所以给出了另外的解决方案,就是我下面说的第二种方式。</p>
<p>第二种方式需要用到数据库,先来建立一个定时任务表并插入三条定时任务记录:</p>
<div><pre><code>drop table if exists spring_scheduled_cron;
create table spring_scheduled_cron (
cron_id int primary key auto_increment
comment '主键id',
cron_key varchar(128) not null unique
comment '定时任务完整类名',
cron_expression varchar(20) not null
comment 'cron表达式',
task_explain varchar(50) not null default ''
comment '任务描述',
status tinyint not null default 1
comment '状态,1:正常;2:停用',
unique index cron_key_unique_idx(cron_key)
)
ENGINE = InnoDB
DEFAULT CHARSET = utf8mb4
COMMENT = '定时任务表';

insert into spring_scheduled_cron
values (1, 'org.javamaster.b2c.scheduled.task.DynamicPrintTask', '/5 * * * * ?', '定时任务描述', 1);
insert into spring_scheduled_cron
values (2, 'org.javamaster.b2c.scheduled.task.DynamicPrintTask1', '
/5 * * * * ?', '定时任务描述1', 1);
insert into spring_scheduled_cron
values (3, 'org.javamaster.b2c.scheduled.task.DynamicPrintTask2', '/5 * * * * ?', '定时任务描述2', 1);</code></pre></div>
<p>编写一个配置类:</p>
<div><pre><code>@Configuration
public class ScheduledConfig implements SchedulingConfigurer {
@Autowired
private ApplicationContext context;
@Autowired
private SpringScheduledCronRepository cronRepository;
@Override
public void configureTasks(ScheduledTaskRegistrar taskRegistrar) {
for (SpringScheduledCron springScheduledCron : cronRepository.findAll()) {
Class<?> clazz;
Object task;
try {
clazz = Class.forName(springScheduledCron.getCronKey());
task = context.getBean(clazz);
} catch (ClassNotFoundException e) {
throw new IllegalArgumentException("spring_scheduled_cron表数据" + springScheduledCron.getCronKey() + "有误", e);
} catch (BeansException e) {
throw new IllegalArgumentException(springScheduledCron.getCronKey() + "未纳入到spring管理", e);
}
Assert.isAssignable(ScheduledOfTask.class, task.getClass(), "定时任务类必须实现ScheduledOfTask接口");
// 可以通过改变数据库数据进而实现动态改变执行周期
taskRegistrar.addTriggerTask(((Runnable) task),
triggerContext -> {
String cronExpression = cronRepository.findByCronKey(springScheduledCron.getCronKey()).getCronExpression();
return new CronTrigger(cronExpression).nextExecutionTime(triggerContext);
}
);
}
}
@Bean
public Executor taskExecutor() {
return Executors.newScheduledThreadPool(10);
}
}</code></pre></div>
<p>这里我为了做到可以灵活处理,自定义了一个接口ScheduledOfTask:</p>
<div><pre><code>/
*

}</code></pre></div>
<p>项目跑起来后,可以看到类被定时执行了:</p>

<p>那么,要如何动态改变执行周期呢,没有理由去手工改动数据库吧?开发测试环境可以这么搞,生产环境就不可以了,所以为了做到动态改变数据库数据,很简单,提供一个Controller类供调用:</p>
<div><pre><code>/**

<p>网页效果是这样的:</p>

<p>可以做到查看任务列表,修改任务cron表达式(也就实现了动态改变定时任务执行周期),暂停定时任务,以及直接执行定时任务。</p>

                    </div><blockquote><p>本文使用 <a href="https://www.jianshu.com/p/5709df6fb58d" class="internal">文章同步助手</a> 同步</p></blockquote>
上一篇 下一篇

猜你喜欢

热点阅读