1. redisson源码剖析-公平锁之排队加锁原理

公平锁加锁的源码在RedissonFairLock

对于lua脚本中的一些参数值的说明,因为lua脚本中设计到很多的参数,提交的提取出来,方便看lua脚本的时候进行分析

KEYS = Arrays.<Object>asList(getName(), threadsQueueName, timeoutSetName)
KEYS[1] = getName() = 锁的名字,“anyLock”
KEYS[2] = threadsQueueName = redisson_lock_queue:{anyLock},基于redis的数据结构实现的一个队列
KEYS[3] = timeoutSetName = redisson_lock_timeout:{anyLock},基于redis的数据结构实现的一个Set数据集合,有序集合,可以自动按照你给每个数据指定的一个分数(score)来进行排序

ARGV =  internalLockLeaseTime, getLockName(threadId), currentTime + threadWaitTime, currentTime
ARGV[1] = 30000毫秒
ARGV[2] = UUID:threadId
ARGV[3] = 当前时间(10:00:00) + 5000毫秒 = 10:00:05
ARGV[4] = 当前时间(10:00:00)

lua脚本中的1、2、3 分别代表第一个、第二个、第三个客户端,代表加锁的顺序

@Override
<T> RFuture<T> tryLockInnerAsync(long leaseTime, TimeUnit unit, long threadId, RedisStrictCommand<T> command) {
    internalLockLeaseTime = unit.toMillis(leaseTime);


    long currentTime = System.currentTimeMillis();
    if (command == RedisCommands.EVAL_NULL_BOOLEAN) {
        return commandExecutor.evalWriteAsync(getName(), LongCodec.INSTANCE, command,
                // remove stale threads
                "while true do "
                + "local firstThreadId2 = redis.call('lindex', KEYS[2], 0);"
                + "if firstThreadId2 == false then "
                    + "break;"
                + "end; "
                + "local timeout = tonumber(redis.call('zscore', KEYS[3], firstThreadId2));"
                + "if timeout <= tonumber(ARGV[3]) then "
                    + "redis.call('zrem', KEYS[3], firstThreadId2); "
                    + "redis.call('lpop', KEYS[2]); "
                + "else "
                    + "break;"
                + "end; "
              + "end;"
                + 
                
                "if (redis.call('exists', KEYS[1]) == 0) and ((redis.call('exists', KEYS[2]) == 0) "
                        + "or (redis.call('lindex', KEYS[2], 0) == ARGV[2])) then " +
                        "redis.call('lpop', KEYS[2]); " +
                        "redis.call('zrem', KEYS[3], ARGV[2]); " +
                        "redis.call('hset', KEYS[1], ARGV[2], 1); " +
                        "redis.call('pexpire', KEYS[1], ARGV[1]); " +
                        "return nil; " +
                    "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]); " +
                        "return nil; " +
                    "end; " +
                    "return 1;", 
                Arrays.<Object>asList(getName(), threadsQueueName, timeoutSetName), 
                internalLockLeaseTime, getLockName(threadId), currentTime);
    }
    
    // 公平锁源码入口
    if (command == RedisCommands.EVAL_LONG) {
        return commandExecutor.evalWriteAsync(getName(), LongCodec.INSTANCE, command,
                // remove stale threads
                // 1. 第一次,如果某个锁,没有人加锁,此时第一个客户端进来,先进入一个死循环
                // 2. 第二个客户端进来尝试加锁,进入死循环中
                // 3. 第三个客户端过来加锁,进入while true 的死循环当中
                "while true do “
                // 1. 第一次进来,这个lindex redisson_lock_queue:{anyLock} 0 的命令就是说,从 redisson_lock_queue:{lock}这个队列中弹出第一个元素,刚开始,肯定是空的,什么都没有,直接就会直接
                // break掉,跳出死循环
                // 2. 第二个客户端首先执行 lindex redisson_lock_queue:{anyLock} 0,取出队列的第一个元素,此时该队列还是空的,然后break掉,跳出死循环
                // 3. lindex redisson_lock_queue:{anyLock} 0,取出队列的第一个元素,此时,这个队列中由于第二个客户端已经将数据插入到队列中,已经在排队了,所以值是有的firstThreadId2=10:00:25
                + "local firstThreadId2 = redis.call('lindex', KEYS[2], 0);"
                + "if firstThreadId2 == false then "
                    + "break;"
                + "end; “
                // 3. zscore redisson_lock_timeout:{anyLock} UUID_02:threadId_02,从有序集合中获取UUID_02:threadId_02对应的分数10:00:25
                + "local timeout = tonumber(redis.call('zscore', KEYS[3], firstThreadId2));”
                // 3. 如果第三个客户端尝试加锁的时间是10:00:05,timeout < 10:00:05,如果条件成立,跳出死循环,如果不成立,具体逻辑后面的笔记中进行分析
                + "if timeout <= tonumber(ARGV[4]) then "
                    + "redis.call('zrem', KEYS[3], firstThreadId2); "
                    + "redis.call('lpop', KEYS[2]); "
                + "else "
                    + "break;"
                + "end; "
              + "end;"
                    // 1. exists anyLock,锁不存在,也就是没人加锁,刚开始因为第一个客户端进来,之前肯定是没有人加锁的,这个条件是成立的
                    // 1. exists redisson_lock_queue:{anyLock}这个队列不存在,或者lindex redisson_lock_queue:{anyLock} 0 这个队列中的第一个元素是UUID:threadId ,
                    // 这个队列存在,但是排在这个队列的第一个元素是当前线程,那么此时这个条件就会成立。
                    // 第一个客户端进来的时候,其实anyLock和队列redisson_lock_queue:{anyLock}都是不存在的,所以条件是成立的

                    // 2. exists anyLock 是否存在,这个时候肯定已经存在了,所以这里的if 判断条件不成立
                    // 3. 第三个客户端进行条件判断,和第二个客户端是一样的,条件不成立
                  + "if (redis.call('exists', KEYS[1]) == 0) and ((redis.call('exists', KEYS[2]) == 0) "
                        + "or (redis.call('lindex', KEYS[2], 0) == ARGV[2])) then “ +
                        // 1. lpop redisson_lock_queue:{anylock} 弹出第一个元素,,队列是空的,所以第一个客户端进来的时候,这行命令是什么都不会干的 
                        "redis.call('lpop', KEYS[2]); “ +
                        // 1. zrem redisson_lock_timeout:{anyLock} UUID_01:threadId,从set集合中删除threadId对应的元素,此时set集合是空的,所以什么都不会做
                        "redis.call('zrem', KEYS[3], ARGV[2]); “ +
                        // 1. hset anyLock uuid:threadId 1 进行加锁,数据结构就是 anyLock:{“UUID_01:threadId_01” : 1 }
                        "redis.call('hset', KEYS[1], ARGV[2], 1); “ +
                        // 1. Pexpire anyLock 30000 ,将anyLock这个key的生存周期设置为3000毫秒(30秒)
                        "redis.call('pexpire', KEYS[1], ARGV[1]); “ +
                        // 1. 返回一个nil,在外层代码中,就会认为是加锁成功,此时就会开启一个watchdog看门狗定时调度的程序,
                        // 每隔10秒判断一下,当前这个线程是否还对这个锁key持有着锁,如果是,则刷新锁key的生存时间为30000毫秒
                        "return nil; " +
                    "end; “ +
                    // 2. 第二个客户端会进入到这段逻辑中,hexists anyLock UUID_02:threadId_02是否存在,条件也不成立
                    // 3. 第三个客户端进行这段逻辑,hexists anyLock UUID_03:threadId_03是否存在,条件也不成立
                    "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]); " +
                        "return nil; " +
                    "end; " +
                        
                    // 2. 第二个客户端会执行如下的逻辑,lindex redisson_lock_queue:{anyLock} 0 从队列中获取第一个元素,肯定是空的
                    // 3. 第三个客户端执行lindex redisson_lock_queue:{anyLock} 0,从队列中取出第一个元素,,这个是有值的,为UUID_02:thread_02
                    "local firstThreadId = redis.call('lindex', KEYS[2], 0); “ +
                    //  取当前的时间
                    "local ttl; “ + 
                    // 2. 判断firstThreadId 不等于空and  firstThreadId=UUID_02:threadId_02条件不成立,走else逻辑
                    // 3.  判断firstThreadId 不等于空and  firstThreadId!=UUID_03:threadId_03条件成立,条件是成立的,其实就是判断在队列中排队的第一个元素是不是当前客户端
                    "if firstThreadId ~= false and firstThreadId ~= ARGV[2] then “ + 
                        // 3. Zscore redisson_lock_timeout:{anyLock} UUID_02:threadId_02 = 10:00:25  - 10:00:05 = 20000毫秒
                        "ttl = tonumber(redis.call('zscore', KEYS[3], firstThreadId)) - tonumber(ARGV[4]);" + 
                    "else “
                    // 2. ttl = pttl anyLock,为anyLock的剩余生存时间,假设当前剩余20000毫秒 
                    // 3. ttl = 20000毫秒
                      + "ttl = redis.call('pttl', KEYS[1]);" + 
                    "end; “ + 
                    // 假设当前时间为10:00:00 
                     // 2. timeout = ttl + 当前时间 + 50000  =   20000 + :10:00:00 + 5000 = 10:00:25
                    // 3. timeout = 20000毫秒 + 10:00:05 + 5000毫秒 = 10:00:30
                    "local timeout = ttl + tonumber(ARGV[3]);” + 
                    // 2. zadd rediss_lock_timeout:{anyLock} 10:00:25 UUID_02:threadId_02,在set集合中插入一个元素,元素的值是UUID_02:threadId_02
                    // 2. 他对应的分数应该是10:00:25(会用这个时间对应的时间戳,也就是long类型表示这个时间,时间越靠后值就越大),而sort set是一个有序集合,会
                    // 2. 会根据这个分数进行排序
                    // 3.zadd redisson_lock_timeout:{anyLock} 10:00:30 UUID_03:threadId_03
                    "if redis.call('zadd', KEYS[3], timeout, ARGV[2]) == 1 then “ +
                    // 2. rpush redisson_lock_queue:{anyLock} UUID_02:threadId_02 ,将UUID_02:threadId02插入到这个队列中 
                    // 3. rpush redisson_lock_queue:{anyLock} UUID_03:theadId_03
                        "redis.call('rpush', KEYS[2], ARGV[2]);" +
                    "end; “ +
                    // 2. 返回ttl,就是anyLock的剩余生存时间,如果拿到的ttl是一个数字的话,那么第二个客户端就会进入死循环,每隔一段时间过来进行尝试加锁,重新执行这段lua脚本
                    "return ttl;", 
                    Arrays.<Object>asList(getName(), threadsQueueName, timeoutSetName), 
                                internalLockLeaseTime, getLockName(threadId), currentTime + threadWaitTime, currentTime);
    }
    
    throw new IllegalArgumentException();
}
02_公平锁原理.png
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 159,117评论 4 362
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 67,328评论 1 293
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 108,839评论 0 243
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 44,007评论 0 206
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 52,384评论 3 287
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 40,629评论 1 219
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 31,880评论 2 313
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 30,593评论 0 198
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 34,313评论 1 243
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 30,575评论 2 246
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 32,066评论 1 260
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 28,392评论 2 253
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 33,052评论 3 236
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 26,082评论 0 8
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 26,844评论 0 195
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 35,662评论 2 274
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 35,575评论 2 270