【Redisson】锁可重入原理

目录

一、基本原理

二、源码解析:

(2)获取锁

(1)释放锁:


之前给大家介绍过redisson的分布式锁,用redisson来实现比自己手搓简单的分布式锁有很多好处,因为这些可重入、可重试的逻辑较为复杂,他们的lua脚本不可能让我们在开发时自己实现,这样太耗时间了。所以redisson就帮我们实现了可重入和可重试。但我们也要知道他的原理,这期我们先讲可重入。


上一篇:

【Redisson】快速实现分布式锁-CSDN博客


一、基本原理

基本原理:

(1)获取锁的时候,还是使用setnx命令,如果该锁还没有被获取过,直接set,也就是获取锁成功

(2)当该线程第二次获取锁时,因为锁已经被获取了,也就是setnx返回false。这时候,先不直接返回获取锁失败的逻辑,而是先判断该锁的value值是不是自己,如果是,那么就获取锁成功。

(3)有一个计数器,每次获取锁成功计数器就+1,每次释放锁的时候计数器就-1。当计数器为0的时候表明该线程所有重入的锁都被释放完毕了,就可以删除这个key。那么其他线程就可以获取锁了

(4)因为key的value值既要记录线程标识,也要记录重入次数。所以String类型的结构就不满足了。需要使用到hash结构

(5)注意:每次释放锁时,如果value值不是0,说明该锁还是被该线程占用的。也就是释放重入锁的时候,要重置锁的有效期

图解:



二、源码解析:

(2)获取锁

java 复制代码
<T> RFuture<T> tryLockInnerAsync(long waitTime, long leaseTime, TimeUnit unit, long threadId, RedisStrictCommand<T> command) {
            return this.commandExecutor.syncedEvalNoRetry(this.getRawName(), LongCodec.INSTANCE, command, 
                    "local mode = redis.call('hget', KEYS[1], 'mode');" +
                    "   if (mode == false) then " +
                    "       redis.call('hset', KEYS[1], 'mode', 'read');" +
                    "       redis.call('hset', KEYS[1], ARGV[2], 1);" +
                    "       redis.call('set', KEYS[2] .. ':1', 1);" +
                    "        redis.call('pexpire', KEYS[2] .. ':1', ARGV[1]);" +
                    "       redis.call('pexpire', KEYS[1], ARGV[1]);" +
                    "       return nil;" +
                    "   end;" +
                    "   if (mode == 'read') or (mode == 'write' and redis.call('hexists', KEYS[1], ARGV[3]) == 1) then" +
                    "        local ind = redis.call('hincrby', KEYS[1], ARGV[2], 1);" +
                    "        local key = KEYS[2] .. ':' .. ind;redis.call('set', key, 1);" +
                    "        redis.call('pexpire', key, ARGV[1]);" +
                    "        local remainTime = redis.call('pttl', KEYS[1]);" +
                    "        redis.call('pexpire', KEYS[1], math.max(remainTime, ARGV[1]));" +
                    "        return nil;" +
                    "   end;" + 
                    "return redis.call('pttl', KEYS[1]);",
                    Arrays.asList(this.getRawName(), this.getReadWriteTimeoutNamePrefix(threadId)), new Object[]{unit.toMillis(leaseTime), this.getLockName(threadId), this.getWriteLockName(threadId)});
        }

解析:

(1)释放锁:

解析:

相关推荐
phltxy12 分钟前
Redis Hash 数据类型:详解命令与实战场景
redis·算法·哈希算法
我是唐青枫8 小时前
终于不用手搓两级缓存了!C#.NET HybridCache 详解:L1 L2、标签失效与防击穿实战
redis·缓存·c#·.net
.柒宇.14 小时前
Redis主从复制集群搭建详解
数据库·redis·缓存·主从复制
IT策士16 小时前
Python 中间件系列:redis 深入浅出
redis·python·中间件
小猿姐16 小时前
GitLab on Kubernetes:使用 KubeBlocks 部署生产级高可用 PostgreSQL 和 Redis
redis·postgresql·kubernetes
phltxy17 小时前
Redis 常见数据类型之全局通用命令详解
数据库·redis·bootstrap
難釋懷17 小时前
Redis网络模型-用户空间和内核态空间
网络·arm开发·redis
薪火铺子17 小时前
布隆过滤器原理与 Redis 防穿透实战
数据库·redis·缓存
.柒宇.18 小时前
Redis高频面试题与跳跃表原理详解
数据库·redis·缓存
未若君雅裁19 小时前
Redis 和 MySQL 双写一致性:延迟双删、读写锁、MQ、Canal 怎么选?
数据库·redis·面试