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():校验当前线程是否持有锁,防止释放别人的锁抛异常

相关推荐
Chase_______1 小时前
【Java基础 | 11】异常处理进阶:throw、throws、自定义异常与异常链讲清楚
java·开发语言·python
cheems95271 小时前
[Redis] redis常见命令和String字符串解析
数据库·redis·缓存
DFT计算杂谈1 小时前
VASP 磁性结构可视化:一键生成完美 VESTA / MCIF
java·前端·css·html·css3
砍材农夫2 小时前
物联网实战:Spring Boot MQTT | 模拟器Paho客户端拆解核心点
java·javascript·网络·spring boot·后端·物联网
weixin_539446782 小时前
使用Java HttpServletResponse和JavaScript Fetch下载文件
java·javascript·python
我登哥MVP2 小时前
Spring Boot 从“会用”到“精通”:自动装配原理
java·spring boot·后端·spring·tomcat·maven·intellij-idea
小的~~2 小时前
Java线程及线程池的相关的问题
java·开发语言·多线程
爱吃羊的老虎2 小时前
【JAVA】Java微服务—网关Gateway
java·微服务·gateway
Jul1en_2 小时前
【Redis】一文讲透缓存更新策略与缓存预热、穿透、雪崩、击穿
数据库·redis·缓存