使用方式

  1. SET KEY VALUE TIME NX
  2. DEL KEY

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

存在问题及解决方法

  1. A加锁, B释放
    方法: Redisson 在tryLock时

    1. long threadId = Thread.currentThread().getId();
    2. protected String getLockName(long threadId) {
    3. return id + ":" + threadId;
    4. }
    5. // id 为 UUID
  2. 会将当前 uuId+线程id写入到锁信息中, unlock时会校验是否是当前线程

  3. A lock锁住之后, 设置了时间, 但是在时间内未完成, 导致锁自动释放, 然后B获取锁同时进行操作
    方法: Redisson 在lock时会启动异步线程, 自动延期, 时间为 lockWatchdogTimeout(默认30s)

    1. Timeout task = commandExecutor.getConnectionManager().newTimeout(new TimerTask() {
    2. 省略...
    3. }, internalLockLeaseTime / 3, TimeUnit.MILLISECONDS);
  4. 看源码是延时 1/3的时间后开始, 就是每次1/3时间的时候延期一次. 这样理解不知道对不对

  5. 主从下, A 加锁 Master 成功后未同步给Slave 便宕机, 导致 B发现未加锁
    方法: 可以修改源码, 同时加锁Master-Slave 才算加锁成功
  6. 集群状态下可以参考RedLock(红锁), 加锁多台机器, 多数成功才算成功(locks.size()/2 + 1)
    1. public class RedissonRedLock extends RedissonMultiLock {
    2. public RedissonRedLock(RLock... locks) {
    3. super(locks);
    4. }
    5. }