Redisson解锁失败,WatchDog会不会一直续期下去?

目录

[一、WatchDog 底层续期规则(Redisson 自动续期锁,只针对lock()无过期时间的锁)](#一、WatchDog 底层续期规则(Redisson 自动续期锁,只针对lock()无过期时间的锁))

二、分场景详解解锁失败

[场景 1:unlock () 抛出异常解锁失败,但业务代码正常执行完毕,线程退出](#场景 1:unlock () 抛出异常解锁失败,但业务代码正常执行完毕,线程退出)

[场景 2:unlock 没执行、业务线程卡死(死循环 / 阻塞 / 休眠永不结束)→ 解锁彻底失败](#场景 2:unlock 没执行、业务线程卡死(死循环 / 阻塞 / 休眠永不结束)→ 解锁彻底失败)

[场景 3:Redis 手动 del 删除锁,代码 unlock 失败](#场景 3:Redis 手动 del 删除锁,代码 unlock 失败)

三、如何避免解锁失败导致无限续期死锁?


解锁正常调用 unlock() 成功 → WatchDog 立刻停止续期;解锁失败(代码抛异常、手动删 key、解锁逻辑报错)分两种场景:

  1. 业务线程正常结束(方法走完 /return):WatchDog 自动停止续期,不会永久续命
  2. 业务线程卡死死循环 / 阻塞永不退出 + unlock 没执行成功:WatchDog 会无限续期,锁永久占用

一、WatchDog 底层续期规则(Redisson 自动续期锁,只针对lock()无过期时间的锁)

只有不带 leaseTime 参数的 RLock.lock() 才开启看门狗续期;lock(10, TimeUnit.SECONDS) 指定过期时间,没有 WatchDog

  • 默认续期间隔:锁过期时间/2 = 15s(默认锁超时 30s),每 15s 异步刷新锁过期为 30s
  • 看门狗依附「持有锁的业务线程生命周期」:业务线程存活 → 定时任务一直跑;线程终止 → 定时任务被取消,续期停止。

二、分场景详解解锁失败

场景 1:unlock () 抛出异常解锁失败,但业务代码正常执行完毕,线程退出

复制代码
RLock lock = redissonClient.getLock("key");
lock.lock();
try {
    // 业务逻辑正常跑完
}finally {
    lock.unlock(); // unlock内部异常,解锁失败
}
  • finally 执行完,当前持有锁的业务线程结束 → WatchDog 的定时续期任务被 cancel ()
  • 锁不再续期,等待30s 自动过期释放,不会死锁。

unlock 源码:解锁无论成功失败,只要线程结束,看门狗任务被注销。

场景 2:unlock 没执行、业务线程卡死(死循环 / 阻塞 / 休眠永不结束)→ 解锁彻底失败

复制代码
lock.lock();
try{
    while(true){ // 线程无限死循环,永不跳出try
        // 死循环卡死
    }
}finally {
    // 代码永远进不到finally,unlock从未执行
}
  1. 线程一直存活,WatchDog 定时任务持续执行
  2. 每 15s 续期锁 30s → 锁无限续命,死锁常驻 Redis

这是线上最容易出现死锁的场景。

场景 3:Redis 手动 del 删除锁,代码 unlock 失败

Redis 客户端手动DEL key把锁删掉:

  • 下次看门狗续期时,setnx 刷新过期失败,直接取消续期任务,不再续约。

三、如何避免解锁失败导致无限续期死锁?

  1. unlock 必须放到 finally 中,保证必定执行解锁逻辑

  2. 业务增加超时保护:子任务设置超时熔断,避免线程永久卡死

  3. 担心 WatchDog 死续期:改用lock(30, TimeUnit.SECONDS)手动指定过期,关闭看门狗,超时自动释放

  4. 捕获 unlock 异常,兜底补偿解锁:

    finally {
    try {
    if(lock.isHeldByCurrentThread()){
    lock.unlock();
    }
    }catch (Exception e){
    // 解锁异常兜底,可发告警
    }
    }

isHeldByCurrentThread():校验当前线程是否持有锁,防止释放别人的锁抛异常

相关推荐
plainGeekDev9 小时前
MVC 写法 → MVVM
android·java·kotlin
SL_staff9 小时前
3周搭完MES系统:JVS低代码+JVS-IoT物联网的实战记录
java·前端·低代码
MacroZheng9 小时前
斩获20w star!Claude Code最强插件,AI编程必备!
java·人工智能·后端
唐青枫11 小时前
Java Spring WebFlux 实战指南:用 Mono、Flux 和 WebClient 写响应式接口
java·spring
小bo波1 天前
使用Thread子类创建线程 VS 使用Runnable接口创建线程的区别
java·多线程·thread·并发编程·runnable
SamDeepThinking1 天前
高并发场景下,CompletableFuture与ForkJoinPool该如何取舍?
java·后端·面试
张不才1 天前
CPU 100% 了怎么办?Java 性能排障的标准化操作
java·后端
shepherd1111 天前
吞吐量提升 10 倍:高并发大批量数据处理任务的架构演进与性能调优
java·后端·架构
plainGeekDev1 天前
单例模式 → object 声明
android·java·kotlin
用户298698530141 天前
Java 实现 Word 文档文本与图片提取的方法
java·后端