1. 背景
定时任务是实际开发中常见的一类功能,例如每天早上凌晨对前一天的注册用户数量、渠道来源进行统计,并以邮件报表的方式发送给相关人员。相信这样的需求,每个开发伙伴都处理过。
你可以使用 Linux 的 Crontab 启动应用程序进行处理,或者直接使用 Spring 的 Schedule 对任务进行调度,还可以使用分布式调度系统,如果 xxl-job 等。相信你已经轻车熟路、习以为常。直到有一天你接到了一个新需求:
1.新建一组任务,周期性的执行指定 SQL 并将结果以邮件的方式发送给特定人群;2.比较方便的对任务进行管理,比如 启动、停止,修改调度周期等;3.动态添加、移除任务,不需要频繁的修改、发布程序;
停顿几分钟,简单思考一下,有哪几种实现思路呢?
本篇文章将从以下几部分进行讨论:
1.Spring Schedule 配置和使用。首先我们将介绍 Demo 的骨架,并基于 Spring-Boot 完成 Schedule 的配置;2.数据库定时轮询方案。使用 Spring Schedule 定时轮询 数据库,并执行相应任务。在执行任务策略中,我们将尝试同步和异步执行两种方案,并对其优缺点进行分析;3.基于 TaskScheduler 动态配置方案。基于数据库 轮询 或 配置中心 两种方案动态的对 Spring TaskScheduler 进行配置,以实现动态管理任务的目的;4.我们进入分布式环境,利用多个冗余节点解决系统高可用问题,同时使用分布式锁保障只会有一个任务同时执行;
2. Spring Schedule
Spring Boot 上的 Schedule 的使用非常简单,无需增加新的依赖,只需简单配置即可。
1.使用 @EnableScheduling 启用 Schedule;2.在要调度的方法上增加 @Scheduled;
首先,我们需要在启动类上添加 @EnableScheduling 注解,该注解将启用 SchedulingConfiguration 配置类帮我们完成最基本的配置。
@SpringBootApplication
@EnableScheduling
public class ConfigurableScheduleDemoApplication {
public static void main(String[] args) {
SpringApplication.run(ConfigurableScheduleDemoApplication.class, args);
}
}
启用Schedule配置之后,在需要被调度的方法上增加 @Scheduled 注解。
@Service
public class SpringScheduleService {
@Autowired
private TaskService taskService;
@Scheduled(fixedDelay = 5 * 1000, initialDelay = 1000)
public void runTask(){
TaskConfig taskConfig = TaskConfig.builder()
.name("Spring Default Schedule")
.build();
this.taskService.runTask(taskConfig);
}
}
runTask 任务延迟 1s 进行初始化,并以 5s 为间隔进行调度。
Scheduled 注解类的详细配置如下:
配置 | 含义 | 样例 |
---|---|---|
cron | linux crontab 表达式 | @Scheduled(cron="*/5 * * * * MON-FRI") 工作日,每 5 s 调度一次 |
fixedDelay | 固定间隔,上次运行结束,与下次启动运行,相隔固定时长 | @Scheduled(fixedDelay=5000) 运行结束后,5S 后启动一次调度 |
fixedDelayString | 与 fixedDelay 一致 | |
fixedRate | 固定周期,前后两次运行相隔固定的时长 | @Scheduled(fixedRate=5000) 前后两个任务,间隔 5 秒 |
fixedRateString | 与 fixedRate 一致 | |
initialDelay | 第一次执行,间隔时间 | @Scheduled(initialDelay=1000, fixedRate=5000) 第一次执行,延时 1 秒,以后以 5 秒为周期进行调度 |
initialDelayString | <