Redis分布式锁的生产问题解决方案

Java进阶训练营学习笔记
课程: Java进阶训练营
老师: 中华石杉
邀请码: 二维码

使用方式

SET KEY VALUE TIME NX
DEL KEY

一般使用 NX, 只有在锁不存在的时候才加锁成功, 设置时间是为了锁永远得不到释放

存在问题及解决方法

  1. A加锁, B释放

    方法: Redisson 在tryLock时

     long threadId = Thread.currentThread().getId();
    
     protected String getLockName(long threadId) {
         return id + ":" + threadId;
     }
    
     // id 为 UUID
    

    会将当前 uuId+线程id写入到锁信息中, unlock时会校验是否是当前线程

  2. A lock锁住之后, 设置了时间, 但是在时间内未完成, 导致锁自动释放, 然后B获取锁同时进行操作

    方法: Redisson 在lock时会启动异步线程, 自动延期, 时间为 lockWatchdogTimeout(默认30s)

     Timeout task = commandExecutor.getConnectionManager().newTimeout(new TimerTask() {
         省略...
     }, internalLockLeaseTime / 3, TimeUnit.MILLISECONDS);
    
    

    看源码是延时 1/3的时间后开始, 就是每次1/3时间的时候延期一次. 这样理解不知道对不对

  3. 主从下, A 加锁 Master 成功后未同步给Slave 便宕机, 导致 B发现未加锁
    方法: 可以修改源码, 同时加锁Master-Slave 才算加锁成功

  4. 集群状态下可以参考RedLock(红锁), 加锁多台机器, 多数成功才算成功(locks.size()/2 + 1)

     public class RedissonRedLock extends RedissonMultiLock {
         public RedissonRedLock(RLock... locks) {
             super(locks);
         }
     }
    

   版权声明

文章作者: liuzhihang
版权声明: 本博客所有文章除特別声明外,均采用 CC BY 4.0 许可协议。转载请注明来源!