Redisson看门狗原理

redission锁原理

  • watchDog 只有在未显示指定加锁时间(leaseTime)时才会生效。(这点很重要)
  • lockWatchdogTimeout设定的时间不要太小 ,比如我之前设置的是 100毫秒,由于网络直接导致加锁完后,watchdog去延期时,这个key在redis中已经被删除了。
  • 在调用lock方法时,会最终调用到tryAcquireAsync。调用链为:lock()->tryAcquire->tryAcquireAsync,详细解释如下:
private <T> RFuture<Long> tryAcquireAsync(long waitTime, long leaseTime, TimeUnit unit, long threadId) {
    
    
    RFuture<Long> ttlRemainingFuture;
    //如果指定了加锁时间,会直接去加锁
    if (leaseTime != -1) {
    
    
        ttlRemainingFuture = tryLockInnerAsync(waitTime, leaseTime, unit, threadId, RedisCommands.EVAL_LONG);
    } else {
    
    
        //没有指定加锁时间 会先进行加锁,并且默认时间就是 LockWatchdogTimeout的时间
        //这个是异步操作 返回RFuture 类似netty中的future
        ttlRemainingFuture = tryLockInnerAsync(waitTime, internalLockLeaseTime,
                                               TimeUnit.MILLISECONDS, threadId, RedisCommands.EVAL_LONG);
    }

    //这里也是类似netty Future 的addListener,在future内容执行完成后执行
    ttlRemainingFuture.onComplete((ttlRemaining, e) -> {
    
    
        if (e != null) {
    
    
            return;
        }

        // lock acquired
        if (ttlRemaining == null) {
    
    
            // leaseTime不为-1时,不会自动延期
            if (leaseTime != -1) {
    
    
                internalLockLeaseTime = unit.toMillis(leaseTime);
            } else {
    
    
                //这里是定时执行 当前锁自动延期的动作,leaseTime为-1时,才会自动延期
                scheduleExpirationRenewal(threadId);
            }
        }
    });
    return ttlRemainingFuture;
}

scheduleExpirationRenewal 中会调用renewExpiration。 这里我们可以看到是启用一个timeout定时,去执行延期动作,

private void renewExpiration() {
    
    
        ExpirationEntry ee = EXPIRATION_RENEWAL_MAP.get(getEntryName());
   // 解锁操作会 EXPIRATION_RENEWAL_MAP.remove(getEntryName());
        if (ee == null) {
    
    
            return;
        }
 
        Timeout task = commandExecutor.getConnectionManager().newTimeout(new TimerTask() {
    
    
            @Override
            public void run(Timeout timeout) throws Exception {
    
    
                ExpirationEntry ent = EXPIRATION_RENEWAL_MAP.get(getEntryName());
                if (ent == null) {
    
    
                    return;
                }
                Long threadId = ent.getFirstThreadId();
                if (threadId == null) {
    
    
                    return;
                }
 
                RFuture<Boolean> future = renewExpirationAsync(threadId);
                future.onComplete((res, e) -> {
    
    
                    if (e != null) {
    
    
                        log.error("Can't update lock " + getRawName() + " expiration", e);
                        EXPIRATION_RENEWAL_MAP.remove(getEntryName());
                        return;
                    }
 
                    if (res) {
    
    
                        //如果 没有报错,就再次定时延期
                        // reschedule itself
                        renewExpiration();
                    } else {
    
    
                        cancelExpirationRenewal(null);
                    }
                });
            }
            // 这里我们可以看到定时任务 是 lockWatchdogTimeout 的1/3时间去执行 renewExpirationAsync
        }, internalLockLeaseTime / 3, TimeUnit.MILLISECONDS);
 
        ee.setTimeout(task);
    }
 protected RFuture<Boolean> renewExpirationAsync(long threadId) {
    
    
        return this.evalWriteAsync(this.getRawName(), LongCodec.INSTANCE, RedisCommands.EVAL_BOOLEAN, "if (redis.call('hexists', KEYS[1], ARGV[2]) == 1) then redis.call('pexpire', KEYS[1], ARGV[1]); return 1; end; return 0;", Collections.singletonList(this.getRawName()), this.internalLockLeaseTime, this.getLockName(threadId));

结论

watch dog 在当前节点存活时每10s给分布式锁的key续期 30s;
watch dog 机制启动,且代码中没有释放锁操作时,watch dog 会不断的给锁续期;
如果程序释放锁操作时因为异常没有被执行,那么锁无法被释放,所以释放锁操作一定要放到 finally {} 中;
要使 watchLog机制生效 ,lock时 不要设置 过期时间
watchlog的延时时间 可以由 lockWatchdogTimeout指定默认延时时间,但是不要设置太小。如100
watchdog 会每 lockWatchdogTimeout/3时间,去延时。
watchdog 通过 类似netty的 Future功能来实现异步延时
watchdog 最终还是通过 lua脚本来进行延时

看门狗的一些思考

看门狗模式简单来说,就是在主线程/服务执行任务运行期间,开启一个守护线程,进行跟踪、进度监控和进行功能补充(延期,告警等),这种模式可以在以后其他需要的地方使用。未完待续。。

猜你喜欢

转载自blog.csdn.net/qq_37436172/article/details/130656960