Spring Cloud Alibaba 集成分布式定时任务调度功能_博客-阿里云Spring Cloud Alibaba官网

Spring Cloud Alibaba 集成分布式定时任务调度功能

发布时间 2024-07-22


背景简介

定时任务是指在约定的时间,或者按照固定频率周期性执行的任务。在企业应用中,非用户行为发起的后台业务,一般都是通过定时任务来实现,常见场景如下:

  • 异步数据处理:比如先将订单入库,每分钟扫描未支付的订单做批处理。
  • 自动化运维:比如每小时清理一次数据库的历史记录。
  • 系统监控:比如每分钟扫描监控指标,如果超出阈值,进行报警。
  • 数据同步:比如每天凌晨1点把mysql中的数据同步到大数据平台中。

在单体应用中,实现定时任务很简单,比如Java中有java.util.Timer和ScheduledExecutorService。在Spring框架中也提供了scheduling�方便的开发定时任务。但是在微服务中,一个应用一般来说有很多节点,通过以上实现方式,就会带来一个致命的问题,那就是任务重复执行。

Spring Cloud Alibaba 发布了Scheduling任务调度模块 [#3732] 提供了一套开源、轻量级、高可用的定时任务解决方案,帮助您快速开发微服务体系下的分布式定时任务。

微服务下的分布式任务能力

定时任务功能作为应用开发中重要能力之一,在Spring Framework中提供了对定时任务支持,在其最基础的使用场景下可方便的通过@Scheduled注解轻松的开发出一个定时任务【剖析参考】;也可以通过扩展集成Quartz框架来实现部分分布式能力,但其目前无法方便的通过spring的@Scheduled注解快速地定义一个任务。框架原生提供的这些能力在单体应用场景下够用,但在微服务分布式集群场景下就显得有些力所不能及了。
因此在Spring Cloud Alibaba体系中我们希望提供一个模块能力,能在微服务场景下快速接入使用分布式定时任务能力。分布式定时任务应该具备哪些能力特性,参考如下对比。

在微服务分布式场景中,除了基础的定时任务运行能力,还需提供如:防重复执行、分片运行、任务执行并行度控制等能力,后续还可在此基础上继续探索定时任务对微服务应用服务本身的定时调度支持,欢迎在社区提出需求和建议。

快速接入体验

在新版的Spring Cloud Alibaba版本中提供分布式任务调度快速启动模块spring-cloud-starter-alibaba-schedulerx来支持各类分布式定时任务方案集成。对于采用了Spring Cloud Alibaba的微服务应用可以通过添加如下pom依赖完成接入。

<dependency>
<groupId>com.alibaba.cloud</groupId>
<artifactId>spring-cloud-starter-alibaba-schedulerx</artifactId>
<version>{version}</version>
</dependency>

在应用配置文件提供了spring.cloud.scheduling.distributed-mode参数让用户选择采用哪一种实现方案,目前提供了基于“阿里云schedulerx”和“开源shedlock”两种模式。后续可根据社区用户需求继续增加其他方案的接入。更为具体详细的接入说明可参考:Example示例工程

采用阿里云SchedulerX

接入Spring分布式定时任务,最简单的方式就是直接接入阿里云托管的分布式任务调度平台SchedulerX,不但完全兼容Spring @Scheduled 注解,还具有高可用、高安全、高性能、免运维、低成本等特点。
任务调度平台SchedulerX的任务管理平台,可以帮助您动态新增、修改、运维定时任务,还有报警监控、历史记录、日志服务、链路追踪等企业级可观测方案,同时还可获得如任务分片执行等增强能力,为应用线上定时任务稳定运行保驾护航。

接入步骤

  1. 登录SchedulerX控制台,免费开通服务。
  2. 参考接入文档接入,每个账号拥有5个免费任务额度,可先0成本试用体验。

image.png

采用本地开源方式

不借助云产品,本地接入方式可以再结合 @SchedulerLock 注解来实现分布式定时任务。
@SchedulerLock 注解是一个分布式锁的框架,结合@Scheduled 注解,可以保证任务同一时间,在多个节点上只会执行一次。该框架支持多种分布式锁的实现,比如Jdbc、Zookeeper、Redis等,目前的集成模块中只提供了Jdbc的方案,后续会继续新增其他模式来满足用户现有场景架构所需。其原理如下

接入步骤
下面以Mysql分布式锁为例,演示分布式任务的接入过程,首先需要按本章节开始说明添加spring-cloud-starter-alibaba-schedulerx和对应配置参数,除此之外用户还需完成如下配置:

  1. 由于采用jdbc作为分布式锁实现,因此还需如下pom依赖
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-jdbc</artifactId>
</dependency>
<dependency>
<groupId>mysql</groupId>
<artifactId>mysql-connector-java</artifactId>
<version>8.0.33</version>
</dependency>
  1. 在application.properties文件中添加jdbc配置(如果已经添加可以忽略这步)
spring.datasource.url=jdbc:mysql://127.0.0.1:3306/testdb?useUnicode=true&characterEncoding=utf-8&useSSL=false
spring.datasource.username=root
spring.datasource.password=123456
spring.datasource.driver-class=com.mysql.cj.jdbc.Driver

完成上述配置启动应用后,会在对应数据库中创建一个shedlock表用于定时任务的执行同步处理。
应用程序代码及任务示例参考如下:
配置启动类,启用SchedulingTasks

@SpringBootApplication
@EnableScheduling
@EnableSchedulerLock(defaultLockAtMostFor = "3m")
public class MyApplication {
public static void main(String[] args) {
SpringApplication.run(MyApplication.class, args);
}
}

创建定时任务

import org.joda.time.DateTime;
import org.springframework.scheduling.annotation.Scheduled;
import org.springframework.stereotype.Component;
import net.javacrumbs.shedlock.spring.annotation.SchedulerLock;
@Component
public class SpringJob {
/**
* 每5分钟跑一次
*/
@Scheduled(cron = "0 */5 * * * ?")
@SchedulerLock(name = "SpringJob.job1", lockAtMostFor = "2m", lockAtLeastFor = "1m")
public void job1() {
System.out.println("time=" + DateTime.now().toString("YYYY-MM-dd HH:mm:ss") + " do job1...");
}
/**
* 每5秒跑一次
*/
@Scheduled(fixedRate = 5000)
@SchedulerLock(name = "SpringJob.job2", lockAtMostFor = "4s", lockAtLeastFor = "4s")
public void job2() {
System.out.println("time=" + DateTime.now().toString("YYYY-MM-dd HH:mm:ss") + " do job2...");
}
/**
* 上次跑完之后隔5秒再跑
* @throws InterruptedException
*/
@Scheduled(fixedDelay = 5000)
@SchedulerLock(name = "SpringJob.job3", lockAtMostFor = "4s", lockAtLeastFor = "4s")
public void job3() throws InterruptedException {
System.out.println("time=" + DateTime.now().toString("YYYY-MM-dd HH:mm:ss") + " do job3...");
Thread.sleep(10000);
}
}

未来规划

Spring Cloud Alibaba分布式定时调度模块为微服务应用场景提供了快速集成接入解决方案,后续可为更多分布式定时任务实现方案提供快速集成,比如:继续集成quartz/xxljob等开源组件,且让上述开源组件也都能来适配spring的@Scheduled注解,实现一份任务代码可对接不同调度服务;同时也会为分布式任务场景下所需的业务能力提供扩展支持。欢迎大家给开源社区提出关于分布式定时任务的业务功能诉求和建议。