Redisson lua脚本解读

Redisson锁的redis数据结构

lock_name作为key,

value为hash,hash最多只会有一条数据hash的key为uuid+:+threadId,value为重入次数

uuid:每个RedissonClient里面连接管理中的id=uuid,创建时初始化

Redisson lua脚本解读_第1张图片

加锁lua脚本(copy from version 3.13.6)

KEYS[1] 锁的名字

ARGV[1] 锁自动失效时间(毫秒,默认30s(看门狗续期时长))

ARGV[2] hash子项的key(uuid+threadId)

--如果锁不存在
if (redis.call('exists', KEYS[1]) == 0) then
--重入次数初始为0后加一
redis.call('hincrby', KEYS[1], ARGV[2], 1);
--设锁的过期时间
redis.call('pexpire', KEYS[1], ARGV[1]);
--返回null-代表加锁成功
return nil;
--if结束符
end;
--如果加锁的进程已存在
if (redis.call('hexists', KEYS[1], ARGV[2]) == 1) then
--重入次数加一
redis.call('hincrby', KEYS[1], ARGV[2], 1);
--更新锁的过期时间(毫秒)
redis.call('pexpire', KEYS[1], ARGV[1]);
--返回null-代表重入成功
return nil;
--if结束符
end;
--返回锁的剩余时间(毫秒)-代表加锁失败
return redis.call('pttl', KEYS[1]);

客户端接收到返回值为毫秒值(锁的失效时间),则会进行循环判断,不停的尝试获取

解锁lua脚本

KEYS[1] 锁的名字

KEYS[2] 发布订阅的channel=redisson_lock__channel:{lock_name}

ARGV[1] 发布订阅中解锁消息=0

ARGV[2] 看门狗续期时间

ARGV[3] hash子项的key=uuid+threadId

--如果锁不存在
if (redis.call('hexists', KEYS[1], ARGV[3]) == 0) then
--返回null-代表解锁成功
return nil;
end;
--重入次数减一
local counter = redis.call('hincrby', KEYS[1], ARGV[3], -1);
--如果重入次数不为0,对锁进行续期(使用看门狗的续期时间,默认续期30s)
if (counter > 0) then
redis.call('pexpire', KEYS[1], ARGV[2]);
--返回0-代表锁的重入次数减一,解锁成功
return 0;
--否则重入次数<=0
else
--删除key
redis.call('del', KEYS[1]);
--向channel中发布删除key的消息
redis.call('publish', KEYS[2], ARGV[1]);
--返回1-代表锁被删除,解锁成功
return 1;
end;
return nil;

看门狗续期lua脚本

io.netty.util.TimerTask每10秒执行一次(30(续期时间)/3)

KEYS[1] 锁的名字

ARGV[1]

--自己加的锁存在
if (redis.call('hexists', KEYS[1], ARGV[2]) == 1) then
--续期
redis.call('pexpire', KEYS[1], ARGV[1]);
--1代表续期成功
return 1;
end;
--自己加的锁不存在,后续不需要再续期
return 0;

 

你可能感兴趣的:(源码学习,lua,redisson)