共计 6142 个字符,预计需要花费 16 分钟才能阅读完成。
by emanjusaka from https://www.emanjusaka.top/2024/03/redisson-distributed-lock 彼岸花开可奈何
本文欢送分享与聚合,全文转载请留下原文地址。
实现分布式锁通常有三种形式:数据库、Redis 和 Zookeeper。咱们比拟罕用的是通过 Redis 和 Zookeeper 实现分布式锁。Redisson 框架中封装了通过 Redis 实现的分布式锁,上面咱们剖析一下它的具体实现。
关键点
-
原子性
要么都胜利,要么都失败
-
过期工夫
如果锁还没来得及开释就遇到了服务宕机,就会呈现死锁的问题。给 Redis 的 key 设置过期工夫,即便服务宕机了超过设置的过期工夫锁会主动进行开释。
-
锁续期
因为给锁设置了过期工夫而咱们的业务逻辑具体要执行多长时间可能是变动和不确定的,如果设定了一个固定的过期工夫,可能会导致业务逻辑还没有执行完,锁被开释了的问题。锁续期能保障锁是在业务逻辑执行完才被开释。
-
正确开释锁
保障开释本人持有的锁,不能呈现 A 开释了 B 持有锁的状况。
Redis 实现分布式锁的几种部署形式
-
单机
在这种部署形式中,Redis 的所有实例都部署在同一台服务器上。这种部署形式简单易行,但存在单点故障的危险。如果 Redis 实例宕机,则所有分布式锁都将生效。
-
哨兵
在这种部署形式中,Redis 的多个实例被配置为哨兵。哨兵负责监控 Redis 实例的状态,并在主实例宕机时主动选举一个新的主实例。这种部署形式能够提供更高的可用性和容错性。
-
集群
在这种部署形式中,Redis 的多个实例被配置为一个集群。集群中的每个实例都是平等的,并且能够解决读写操作。这种部署形式能够提供最高的可用性和容错性。
-
红锁
搞几个独立的 Master,比方 5 个,而后挨个加锁,只有超过一半以上(这里是 5/2+1=3 个)就代表加锁胜利,而后开释锁的时候也逐台开释。
应用形式
-
引入依赖
<!-- pom.xml 文件 --> <dependency> <groupId>org.redisson</groupId> <artifactId>redisson-spring-boot-starter</artifactId> <version>3.17.7</version> </dependency>
版本依赖:
redisson-spring-data module name Spring Boot version redisson-spring-data-16 1.3.y redisson-spring-data-17 1.4.y redisson-spring-data-18 1.5.y redisson-spring-data-2x 2.x.y redisson-spring-data-3x 3.x.y
-
yml 配置
spring: redis: redisson: config: singleServerConfig: address: redis://127.0.0.1:6379 database: 0 password: null timeout: 3000
-
间接注入应用
package top.emanjusaka; import org.redisson.api.RLock; import org.redisson.api.RedissonClient; import org.springframework.stereotype.Service; import javax.annotation.Resource; import java.util.concurrent.TimeUnit; /** * @Author emanjusaka www.emanjusaka.top * @Date 2024/2/28 16:41 * @Version 1.0 */ @Service public class Lock { @Resource private RedissonClient redissonClient; public void lock() { // 写入 redis 的 key 值 String lockKey = "lock-test"; // 获取一个 Rlock 锁对象 RLock lock = redissonClient.getLock(lockKey); // 获取锁,并为其设置过期工夫为 10s lock.lock(10, TimeUnit.SECONDS); try { // 执行业务逻辑.... System.out.println("获取锁胜利!"); } finally { // 开释锁 lock.unlock(); System.out.println("开释锁胜利!"); } } }
底层分析
lock()
要害代码
<T> RFuture<T> tryLockInnerAsync(long waitTime, long leaseTime, TimeUnit unit, long threadId, RedisStrictCommand<T> command) {return commandExecutor.syncedEval(getRawName(), LongCodec.INSTANCE, command,
"if ((redis.call('exists', KEYS[1]) == 0)" +
"or (redis.call('hexists', KEYS[1], ARGV[2]) == 1)) then" +
"redis.call('hincrby', KEYS[1], ARGV[2], 1);" +
"redis.call('pexpire', KEYS[1], ARGV[1]);" +
"return nil;" +
"end;" +
"return redis.call('pttl', KEYS[1]);",
Collections.singletonList(getRawName()), unit.toMillis(leaseTime), getLockName(threadId));
}
RFuture<T>
:示意返回一个异步后果对象,其中泛型参数 T 示意后果的类型。-
tryLockInnerAsync
办法承受一下参数:waitTime
:等待时间,用于指定在获取锁时的最大等待时间。leaseTime
:租约工夫,用于指定锁的持有工夫unit
:工夫单位,用于将 leaseTime 转换为毫秒threadId
:线程 ID,用于标识以后线程command
:Redis 命令对象,用于执行 Redis 操作
-
办法体中的代码应用 Lua 脚本来实现分布式锁的逻辑。
-
if ((redis.call(‘exists’, KEYS[1]) == 0) or (redis.call(‘hexists’, KEYS[1], ARGV[2]) == 1)): 如果键不存在或者哈希表中曾经存在对应的线程 ID,则执行以下操作:
- redis.call(‘hincrby’, KEYS[1], ARGV[2], 1): 将哈希表中对应线程 ID 的值加 1。
- redis.call(‘pexpire’, KEYS[1], ARGV[1]): 设置键的过期工夫为租约工夫。
- return nil: 返回 nil 示意胜利获取锁。
-
else: 如果键存在且哈希表中不存在对应的线程 ID,则执行以下操作:
- return redis.call(‘pttl’, KEYS[1]): 返回键的残余生存工夫。
-
commandExecutor.syncedEval
:示意同步执行 Redis 命令LongCodec.INSTANCE
:用于编码和解码长整型数据Collections.singletonList(getRawName())
:创立一个只蕴含一个元素的列表,元素为锁的名称unit.toMillis(leaseTime)
:将租约工夫转换为毫秒getLockName(threadId)
:依据线程 ID 生成锁的名称
// 省去了那些无关重要的代码
private void lock(long leaseTime, TimeUnit unit, boolean interruptibly) throws InterruptedException {long threadId = Thread.currentThread().getId();
// tryAcquire 就是下面剖析的 lua 残缺脚本
Long ttl = tryAcquire(-1, leaseTime, unit, threadId);
// 返回 null 就代表上锁胜利。if (ttl == null) {return;}
// 如果没胜利,也就是锁的剩余时间不是 null 的话,那么就执行上面的逻辑
// 其实就是说 如果有锁(锁剩余时间不是 null),那就死循环期待从新抢锁。try {while (true) {
// 从新抢锁
ttl = tryAcquire(-1, leaseTime, unit, threadId);
// 抢锁胜利就 break 退出循环
if (ttl == null) {break;}
// 省略一些代码
}
} finally {}}
下面代码实现了一个分布式锁的性能。它应用了 Lua 脚本来尝试获取锁,并在胜利获取锁后返回锁的剩余时间(ttl)。如果获取锁失败,则进入一个死循环,一直尝试从新获取锁,直到胜利为止。
unlock()
要害代码
protected RFuture<Boolean> unlockInnerAsync(long threadId) {return evalWriteAsync(getRawName(), LongCodec.INSTANCE, RedisCommands.EVAL_BOOLEAN,
"if (redis.call('hexists', KEYS[1], ARGV[3]) == 0) then" +
"return nil;" +
"end;" +
"local counter = redis.call('hincrby', KEYS[1], ARGV[3], -1);" +
"if (counter > 0) then" +
"redis.call('pexpire', KEYS[1], ARGV[2]);" +
"return 0;" +
"else" +
"redis.call('del', KEYS[1]);" +
"redis.call(ARGV[4], KEYS[2], ARGV[1]);" +
"return 1;" +
"end;" +
"return nil;",
Arrays.asList(getRawName(), getChannelName()),
LockPubSub.UNLOCK_MESSAGE, internalLockLeaseTime, getLockName(threadId), getSubscribeService().getPublishCommand());
}
RFuture<Boolean>
: 示意返回一个异步后果对象,其中泛型参数 Boolean 示意后果的类型。-
unlockInnerAsync
办法承受以下参数:threadId
: 线程 ID,用于标识以后线程。
-
办法体中的代码应用 Lua 脚本来实现分布式锁的解锁逻辑。以下是对 Lua 脚本的解释:
if (redis.call('hexists', KEYS[1], ARGV[3]) == 0)
: 如果哈希表中不存在对应的线程 ID,则返回 nil 示意无奈解锁。local counter = redis.call('hincrby', KEYS[1], ARGV[3], -1)
: 将哈希表中对应线程 ID 的值减 1,并将后果赋值给变量 counter。-
if (counter > 0)
: 如果 counter 大于 0,示意还有其余线程持有锁,执行以下操作:redis.call('pexpire', KEYS[1], ARGV[2])
: 设置键的过期工夫为租约工夫。return 0
: 返回 0 示意锁依然被其余线程持有。
-
else
: 如果 counter 等于 0,示意以后线程是最初一个持有锁的线程,执行以下操作:redis.call('del', KEYS[1])
: 删除键,开释锁。redis.call(ARGV[4], KEYS[2], ARGV[1])
: 调用发布命令,告诉其余线程锁曾经开释。return 1
: 返回 1 示意胜利开释锁。
return nil
: 如果后面的条件都不满足,返回 nil 示意无奈解锁。
evalWriteAsync
办法用于执行 Lua 脚本并返回异步后果对象。getRawName()
: 获取锁的名称。LongCodec.INSTANCE
: 用于编码和解码长整型数据。RedisCommands.EVAL_BOOLEAN
: 指定 Lua 脚本的返回类型为布尔值。Arrays.asList(getRawName(), getChannelName())
: 创立一个蕴含两个元素的列表,元素别离为锁的名称和频道名称。LockPubSub.UNLOCK_MESSAGE
: 公布音讯的内容。internalLockLeaseTime
: 锁的租约工夫。getLockName(threadId)
: 依据线程 ID 生成锁的名称。getSubscribeService().getPublishCommand()
: 获取发布命令。
锁续期
watchDog
外围工作流程是定时监测业务是否执行完结,没完结的话在看你这个锁是不是快到期了(超过锁的三分之一工夫),那就从新续期。这样避免如果业务代码没执行完,锁却过期了所带来的线程不平安问题。
Redisson 的 watchDog 机制底层不是调度线程池,而是间接用的 netty 事件轮。
Redisson 的 WatchDog 机制是用于主动续期分布式锁和监控对象生命周期的一种机制,确保了分布式环境下锁的正确性和资源的及时开释。
- 主动续期:当 Redisson 客户端获取了一个分布式锁后,会启动一个 WatchDog 线程。这个线程负责在锁行将到期时主动续期,保障持有锁的线程能够继续执行工作。默认状况下,锁的初始超时工夫是 30 秒,每 10 秒钟 WatchDog 会查看一次锁的状态,如果锁仍然被持有,它会将锁的过期工夫从新设置为 30 秒。
- 参数配置:能够通过设置 lockWatchdogTimeout 参数来调整 WatchDog 查看锁状态的频率和续期的超时工夫。这个参数默认值是 30000 毫秒(即 30 秒),实用于那些没有明确指定 leaseTimeout 参数的加锁申请。
- 重连机制:除了锁主动续期外,WatchDog 机制还用作 Redisson 客户端的主动重连性能。当客户端与 Redis 服务器失去连贯时,WatchDog 会主动尝试从新连贯,从而复原服务的失常运作。
- 资源管理:WatchDog 也负责监控 Redisson 对象的生命周期,例如分布式锁。当对象的生命周期到期时,WatchDog 会将其从 Redis 中删除,防止过期数据占用过多内存空间。
- 异步加锁:在加锁的过程中,WatchDog 会在 RedissonLock#tryAcquireAsync 办法中发挥作用,该办法是进行异步加锁的逻辑所在。通过这种形式,加锁操作不会阻塞以后线程,进步了零碎的性能。
本文原创,满腹经纶,如有纰漏,欢送斧正。如果本文对您有所帮忙,欢送点赞,并期待您的反馈交换,独特成长。
原文地址:https://www.emanjusaka.top/2024/03/redisson-distributed-lock
微信公众号:emanjusaka 的编程栈