分布式定时任务锁ShedLock
概述
Spring为定时任务提供一个易于实现的API。在没有部署应用程序的多个实例之前,它很有效。默认情况下,Spring无法处理多个实例上的调度程序同步,而是在每个节点上同时执行作业。ShedLock库确保计划任务只能同时运行一次,可以代替Quartz。ShedLock支持Mongo,Redis,Hazelcast,ZooKeeper以及任何基于JDBC驱动程序的数据库。
GitHub
入门
shedlock-spring
基本依赖项:
<dependency>
<groupId>net.javacrumbs.shedlock</groupId>
<artifactId>shedlock-spring</artifactId>
</dependency>
再针对不同的锁,分别添加不同的依赖。如JDBC(数据库)锁添加:
<dependency>
<groupId>net.javacrumbs.shedlock</groupId>
<artifactId>shedlock-provider-jdbc-template</artifactId>
</dependency>
注意JDBC和JdbcTemplate是两种不同的锁。
ShedLock仅适用于具有共享数据库的环境。它在数据库中创建一个表或文档,用于存储有关当前锁的信息。为ShedLock创建一个PostgreSQL数据库表,以保留有关调度程序锁的信息:
CREATE TABLE shedlock(
name VARCHAR(64),
lock_until TIMESTAMP(3) NULL,
locked_at TIMESTAMP(3) NULL,
locked_by VARCHAR(255),
PRIMARY KEY (name)
)
Spring启动类加上@EnableScheduling
和@EnableSchedulerLock
注解:
@SpringBootApplication
@EnableScheduling
@EnableSchedulerLock(defaultLockAtMostFor = "PT30S")
public class SpringApplication {
public static void main(String[] args) {
SpringApplication.run(SpringApplication.class, args);
}
}
defaultLockAtMostFor参数指定在执行节点结束时应保留锁的默认时间量,使用ISO8601 Duration格式。
@Component
public class ShedLockConfig {
@Bean
public LockProvider lockProvider(DataSource dataSource) {
return new JdbcTemplateLockProvider(dataSource);
}
@Bean
public ScheduledLockConfiguration scheduledLockConfiguration(LockProvider lockProvider) {
return ScheduledLockConfigurationBuilder.withLockProvider(lockProvider)
.withPoolSize(10)
.withDefaultLockAtMostFor(Duration.ofMinutes(10))
.build();
}
}
创建任务
方法添加注解@Scheduled
和@SchedulerLock
注释:
@Component
class TaskScheduler {
// 每15分钟
@Scheduled(cron = "0 /15 * * * ?")
@SchedulerLock(name = "scheduledTask", lockAtLeastForString = "PT1M", lockAtMostForString = "PT5M")
public void scheduledTask() {
}
}
@Scheduled 支持cron表达式;
@SchedulerLock注解有五个参数:
- name:定时任务的名字,即数据库中的内个主键,必须唯一
- lockAtMostFor:锁的最大时间单位为毫秒
- lockAtMostForString:最大时间的字符串形式,例如:PT30S 代表30秒
- lockAtLeastFor:锁的最小时间单位为毫秒
- lockAtLeastForString:最小时间的字符串形式
在正常情况下,ShedLock会在任务完成后直接释放锁。@EnableSchedulerLock中提供默认值。
Redis
不建议使用
Redis
@Configuration
@EnableSchedulerLock(defaultLockAtMostFor = "PT1M")
public class ShedLockConfig {
@Bean
public LockProvider lockProvider(RedisTemplate redisTemplate) {
return new RedisLockProvider(redisTemplate.getConnectionFactory());
}
}
@Scheduled(cron = "0 0 */1 * * ?")
@SchedulerLock(name = "job1")
public void job1() {
log.info("Start running job");
}
<dependency>
<groupId>net.javacrumbs.shedlock</groupId>
<artifactId>shedlock-provider-redis-spring</artifactId>
</dependency>
Jedis
<dependency>
<groupId>net.javacrumbs.shedlock</groupId>
<artifactId>shedlock-provider-redis-jedis</artifactId>
</dependency>
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· 开发者必知的日志记录最佳实践
· SQL Server 2025 AI相关能力初探
· Linux系列:如何用 C#调用 C方法造成内存泄露
· AI与.NET技术实操系列(二):开始使用ML.NET
· 记一次.NET内存居高不下排查解决与启示
· Manus重磅发布:全球首款通用AI代理技术深度解析与实战指南
· 被坑几百块钱后,我竟然真的恢复了删除的微信聊天记录!
· 没有Manus邀请码?试试免邀请码的MGX或者开源的OpenManus吧
· 园子的第一款AI主题卫衣上架——"HELLO! HOW CAN I ASSIST YOU TODAY
· 【自荐】一款简洁、开源的在线白板工具 Drawnix