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

使用方式

SET KEY VALUE TIME NX
DEL KEY

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

存在问题及解决方法

  1. A加锁, B释放

    方法: Redisson 在tryLock时

    1
    2
    3
    4
    5
    6
    7
    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)

    1
    2
    3
    4
    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)

    1
    2
    3
    4
    5
    public class RedissonRedLock extends RedissonMultiLock {
    public RedissonRedLock(RLock... locks) {
    super(locks);
    }
    }