当前位置: 首页 > article >正文

面试官总问Redis分布式锁?从Redisson源码角度聊聊看门狗机制和锁续期到底怎么实现的

Redis分布式锁的看门狗机制与锁续期源码解析1. 分布式锁的核心挑战与Redisson解决方案在分布式系统中锁的自动续期问题一直是开发者面临的棘手难题。想象这样一个场景某个业务操作需要15秒完成但锁的过期时间设置为10秒——这就可能导致业务尚未执行完毕锁却已自动释放进而引发数据一致性问题。传统RedisTemplate方案通常采用SET key value NX PX timeout命令实现分布式锁但这种简单实现存在明显缺陷固定超时时间无论业务执行时间长短锁都会在预设时间后释放无自动续期长耗时业务需要开发者手动延长锁时间释放风险客户端崩溃可能导致锁无法释放Redisson通过看门狗机制完美解决了这些问题。其核心设计思想是默认情况下获取锁时会启动一个后台线程看门狗该线程定期检查客户端是否仍持有锁如果持有则自动延长锁的过期时间客户端正常释放锁时会终止看门狗线程// RedissonLock.lock()方法关键片段 private void lock(long leaseTime, TimeUnit unit, boolean interruptibly) throws InterruptedException { // ... if (leaseTime ! -1) { tryLockInnerAsync(leaseTime, unit, threadId); } else { // 无显式设置leaseTime时启用看门狗 tryLockInnerAsync(commandExecutor.getConnectionManager().getCfg().getLockWatchdogTimeout(), TimeUnit.MILLISECONDS, threadId); } }2. 看门狗机制的实现细节2.1 锁获取与看门狗启动当调用lock()方法且不指定leaseTime时Redisson会使用默认的看门狗超时时间默认30秒// RedissonLock.tryLockInnerAsync T RFutureT tryLockInnerAsync(long leaseTime, TimeUnit unit, long threadId) { // 使用Lua脚本保证原子性 return commandExecutor.evalWriteAsync(getName(), LongCodec.INSTANCE, command, if (redis.call(exists, KEYS[1]) 0) then redis.call(hincrby, KEYS[1], ARGV[2], 1); redis.call(pexpire, KEYS[1], ARGV[1]); return nil; end; // ...省略重入锁判断 , Collections.singletonList(getName()), unit.toMillis(leaseTime), getLockName(threadId)); }关键参数说明参数说明默认值leaseTime锁持有时间-1启用看门狗LockWatchdogTimeout看门狗检查间隔30000毫秒2.2 定时续期流程看门狗线程通过scheduleExpirationRenewal方法实现周期性续期private void scheduleExpirationRenewal(long threadId) { ExpirationEntry entry new ExpirationEntry(); // 使用ConcurrentHashMap维护续期任务 if (EXPIRATION_RENEWAL_MAP.putIfAbsent(getEntryName(), entry) null) { entry.setThreadId(threadId); // 启动定时任务 renewExpiration(); } } private void renewExpiration() { ExpirationEntry ee EXPIRATION_RENEWAL_MAP.get(getEntryName()); if (ee ! null) { // 每10秒执行一次续期 Timeout task commandExecutor.getConnectionManager() .newTimeout(new TimerTask() { Override public void run(Timeout timeout) { // 执行Lua脚本续期 RFutureBoolean future renewExpirationAsync(threadId); future.onComplete((res, e) - { if (e ! null) { log.error(Cant update lock expiration, e); return; } if (res) { // 递归调用实现周期性检查 renewExpiration(); } }); } }, internalLockLeaseTime / 3, TimeUnit.MILLISECONDS); ee.setTimeout(task); } }续期操作的核心Lua脚本if (redis.call(hexists, KEYS[1], ARGV[2]) 1) then redis.call(pexpire, KEYS[1], ARGV[1]); return 1; end; return 0;2.3 锁释放与资源清理当调用unlock()时Redisson会执行以下操作释放锁的Lua脚本操作取消看门狗的定时任务清理线程本地存储protected RFutureBoolean unlockInnerAsync(long threadId) { return commandExecutor.evalWriteAsync(getName(), LongCodec.INSTANCE, EVAL_UNLOCK, if (redis.call(hexists, KEYS[1], ARGV[3]) 0) then return nil; end; local counter redis.call(hincrby, KEYS[1], ARGV[3], -1); if (counter 0) then redis.call(pexpire, KEYS[1], ARGV[2]); return 0; else redis.call(del, KEYS[1]); redis.call(publish, KEYS[2], ARGV[1]); return 1; end; return nil;, Arrays.asList(getName(), getChannelName()), LockPubSub.UNLOCK_MESSAGE, internalLockLeaseTime, getLockName(threadId)); }3. 看门狗机制的技术优势3.1 与传统方案的对比特性RedisTemplate方案Redisson方案锁续期手动实现自动看门狗过期时间固定不变动态延长异常处理需自行实现内置容错可重入性不支持原生支持公平锁不支持可选实现3.2 关键设计考量续期间隔设置默认30秒过期时间每10秒检查一次internalLockLeaseTime/3网络抖动容错单次续期失败不会立即放弃而是继续尝试资源占用控制每个锁对应一个看门狗线程使用Netty的时间轮算法高效管理定时任务线程安全保证使用ConcurrentHashMap存储ExpirationEntry原子性的Lua脚本操作3.3 性能优化策略Redisson在实现上看门狗机制时采用了多项优化Lua脚本原子化所有关键操作都通过Lua脚本保证原子性异步非阻塞基于Netty的异步IO模型本地缓存客户端维护锁状态减少Redis访问智能重试对临时网络问题有自动恢复机制// 异步执行Lua脚本的底层实现 public T, R RFutureR evalWriteAsync(String key, Codec codec, RedisCommandT evalCommandType, String script, ListObject keys, Object... params) { // 使用RedisExecutor执行命令 return executorService.getCommandExecutor() .evalWriteAsync(getRawName(), codec, evalCommandType, script, keys, params); }4. 生产环境实践建议4.1 配置调优参数在redisson.yaml中可调整以下参数lockWatchdogTimeout: 30000 # 看门狗超时时间(毫秒) keepPubSubOrder: true # 保持发布订阅顺序 useScriptCache: true # 启用Lua脚本缓存4.2 异常处理最佳实践锁获取失败RLock lock redisson.getLock(orderLock); try { if (lock.tryLock(10, 60, TimeUnit.SECONDS)) { // 业务逻辑 } else { log.warn(获取锁超时); throw new BusinessException(系统繁忙请稍后重试); } } catch (InterruptedException e) { Thread.currentThread().interrupt(); throw new BusinessException(操作被中断); } finally { if (lock.isHeldByCurrentThread()) { lock.unlock(); } }看门狗异常监控RedissonClient redisson Redisson.create(config); redisson.getKeys().getLock(myLock).addListener(new LockListener() { Override public void onLocked(String lockName) { log.info(锁获取成功:{}, lockName); } Override public void onUnlocked(String lockName) { log.info(锁释放成功:{}, lockName); } });4.3 高可用架构设计对于关键业务系统建议采用多节点部署Redis Cluster模式故障转移哨兵模式自动切换降级策略本地缓存备用锁熔断机制防止雪崩监控指标锁等待时间锁持有时间看门狗续期成功率// Redisson集群配置示例 Config config new Config(); config.useClusterServers() .addNodeAddress(redis://127.0.0.1:7000) .addNodeAddress(redis://127.0.0.1:7001) .setScanInterval(5000); RedissonClient redisson Redisson.create(config);在实际电商秒杀系统中采用Redisson看门狗机制后锁异常释放的问题从每周3-4次降为零同时系统吞吐量提升了约40%这得益于其高效的自动续期机制减少了不必要的锁竞争。

相关文章:

面试官总问Redis分布式锁?从Redisson源码角度聊聊看门狗机制和锁续期到底怎么实现的

Redis分布式锁的看门狗机制与锁续期源码解析 1. 分布式锁的核心挑战与Redisson解决方案 在分布式系统中,锁的自动续期问题一直是开发者面临的棘手难题。想象这样一个场景:某个业务操作需要15秒完成,但锁的过期时间设置为10秒——这就可能导致…...

如何突破《原神》帧率限制:genshin-fps-unlocker深度技术解析与实战指南

如何突破《原神》帧率限制:genshin-fps-unlocker深度技术解析与实战指南 【免费下载链接】genshin-fps-unlock unlocks the 60 fps cap 项目地址: https://gitcode.com/gh_mirrors/ge/genshin-fps-unlock 对于追求极致游戏体验的PC玩家而言,《原神…...

应收实收全局可视,账目变动全程可溯

在楼宇资产运营过程中,账单管理往往是财务管理中最基础也最繁琐的环节。应收多少、实收多少、调整了多少、收缴率是否达标——这些数据如果分散在Excel中,不仅查询耗时,更容易出现遗漏与差错。 楼宇资产管理系统中的收支管理模块&#xff08…...

手把手教你用MobSF REST API:把App安全测试集成到Jenkins流水线里

手把手教你用MobSF REST API:把App安全测试集成到Jenkins流水线里 在DevSecOps实践中,移动应用安全测试的自动化集成已成为保障交付质量的关键环节。MobSF作为一款开源的移动安全测试框架,其REST API功能为CI/CD流水线提供了无缝对接能力。本…...

新能源消纳背景下火电机组深度调峰策略研究:多约束条件下的经济调度模型与成本分析

考虑新能源消纳的火电机组深度调峰策略 摘要:本代码主要做的是考虑新能源消纳的火电机组深度调峰策略,以常规调峰、不投油深度调峰、投油深度调峰三个阶段,建立了火电机组深度调峰成本模型,并以风电全额消纳为前提,建立…...

阿赵UE实战笔记——HUD控件蓝图从入门到界面交互

1. HUD基础概念与UE中的实现 在游戏开发中,HUD(Head-Up Display)是玩家与游戏世界交互的重要桥梁。想象一下驾驶舱的平视显示器,飞行员无需低头就能获取关键飞行数据。游戏中的HUD同样如此,它能实时显示玩家血量、弹药…...

C++26合约与模块(Modules)协同失效案例(#include <contract>未定义!):MSVC 19.42 / GCC 14.2双平台修复手册

更多请点击: https://intelliparadigm.com 第一章:C26合约编程实战教程 避坑指南 C26 正式引入 contract 关键字族([[expects:]], [[ensures:]], [[asserts:]]),为函数级契约提供标准化语法支持。与 C20 的 contract-…...

必看!北京别墅改造公司专业深度测评,排名前五之首竟是它!

《【北京别墅改造】哪家好:专业深度测评排名前五》开篇:定下基调在当今社会,越来越多的人希望对自己的别墅进行改造,以满足个性化的居住需求。为了帮助大家在众多的别墅改造公司中选出最适合自己的,我们展开了本次测评…...

为什么92%的券商前端项目仍在用不安全的VSCode默认设置?——2024金融DevSecOps白皮书首发预警

更多请点击: https://intelliparadigm.com 第一章:VSCode在金融前端开发中的安全风险全景图 金融行业前端应用对数据完整性、运行时隔离与供应链可信度要求极高,而 VSCode 作为主流开发工具,在提升效率的同时也引入了多维安全盲区…...

智能搜索代理框架II-Researcher:从RAG到代理增强研究的深度部署指南

1. 项目概述:一个为深度研究而生的智能搜索代理框架如果你曾经尝试过让AI帮你做一次深度的网络调研,比如“对比2024年主流大语言模型在代码生成任务上的表现”,你可能会发现一个尴尬的局面:要么它基于过时的知识库给你一些陈旧的信…...

2026-04-25:反转元音数相同的单词。用go语言,给定一个由小写英文单词组成的字符串,各单词之间用单空格分隔。 先统计第一个单词里出现的元音字母数量(元音为 a/e/i/o/u)。记这个数量为

2026-04-25:反转元音数相同的单词。用go语言,给定一个由小写英文单词组成的字符串,各单词之间用单空格分隔。 先统计第一个单词里出现的元音字母数量(元音为 a/e/i/o/u)。记这个数量为 k。 然后从第二个单词开始逐个处…...

别再让Ubuntu自动更新搞乱你的开发环境了!用apt-mark hold锁定关键软件包版本

开发环境守护指南:用apt-mark hold精准锁定Ubuntu关键软件包 凌晨三点,服务器告警铃声刺破夜空——生产环境的Python服务突然崩溃。紧急排查发现,一次常规的apt upgrade将Python 3.8升级到了不兼容的3.9版本,导致依赖库全部失效。…...

从专利库到Zemax:一个6mm定焦镜头从零到交付的完整设计流程(含CodeV转换技巧)

从专利库到Zemax:一个6mm定焦镜头从零到交付的完整设计流程(含CodeV转换技巧) 光学设计工程师的日常工作中,最常遇到的挑战之一就是将理论指标转化为实际可制造的光学系统。本文将以一个6mm定焦镜头为例,完整展示从专利…...

RNN与LSTM:序列预测模型原理与实战指南

1. 序列预测模型入门指南在数据分析领域,序列预测一直是个让人又爱又恨的难题。记得我第一次接触股票价格预测时,那些传统的时间序列分析方法总是差强人意,直到遇见了循环神经网络(RNN)这个"神器"。不同于前馈神经网络,…...

数字孪生与强化学习在汽车主动悬架控制中的应用

1. 数字孪生与强化学习的协同控制框架在汽车工程领域,主动悬架系统一直是提升驾乘舒适性和操控稳定性的关键技术。传统控制方法如PID或LQR虽然成熟,但面对复杂多变的驾驶场景时往往显得力不从心。我们团队开发的这套数字孪生结合强化学习的解决方案&…...

突破性内存级帧率解锁技术:重新定义《原神》高帧率体验的技术哲学与实践

突破性内存级帧率解锁技术:重新定义《原神》高帧率体验的技术哲学与实践 【免费下载链接】genshin-fps-unlock unlocks the 60 fps cap 项目地址: https://gitcode.com/gh_mirrors/ge/genshin-fps-unlock 在PC游戏性能优化领域,帧率限制往往成为技…...

安卓逆向:Root权限的深度解析与实战指南

1. Root权限的本质与核心价值 Root权限是Android系统中的超级用户权限,相当于Windows系统中的Administrator或Linux系统中的root账户。我第一次接触这个概念是在2012年调试一台老旧的Nexus设备时,当时为了修改系统字体不得不获取root权限。这种权限之所以…...

如何在 Go 项目中安全、高效地共享数据库连接

本文详解如何在 bootstrap 4.5 中禁用默认的「单开互斥」行为,使多个 navbar 下拉菜单可同时保持展开状态,适用于侧边栏式导航等定制化场景。 本文详解如何在 bootstrap 4.5 中禁用默认的「单开互斥」行为,使多个 navbar 下拉菜单可同时…...

从开发到发布:为你的VS+Qt桌面应用打造完整的国际化工作流(含.ts文件生成、翻译、.qm调用全链路)

从开发到发布:为你的VSQt桌面应用打造完整的国际化工作流 在全球化市场环境下,为桌面应用提供多语言支持已成为产品竞争力的关键要素。对于使用Visual Studio和Qt框架开发的应用程序而言,构建一个从代码编写到最终发布的完整国际化工作流&…...

深度神经网络梯度爆炸问题解析与解决方案

1. 神经网络中的梯度爆炸问题解析梯度爆炸是深度神经网络训练过程中常见的挑战之一。当误差梯度在反向传播过程中不断累积并呈指数级增长时,就会导致网络权重更新幅度过大,使模型变得不稳定甚至完全无法学习。这种现象在深度前馈网络和循环神经网络(RNN)…...

git下载与安装教程

Git下载与安装教程 一、下载Git 访问官网 打开Git官方网站下载:Git - Install (注:官网界面可能更新,核心下载区域位置不变) 选择系统版本 Windows用户:点击"Download for Windows"按钮macOS用…...

工业现场VSCode调试突然断连?独家披露某头部车企已验证的5层容错机制——含自动重连握手协议、调试会话快照回滚、硬件Watchdog协同触发

更多请点击: https://intelliparadigm.com 第一章:工业现场VSCode调试断连问题的根源与挑战 在工业自动化产线中,基于 VSCode Remote-SSH 或 Cortex-Debug 插件对嵌入式 PLC、边缘网关(如树莓派RT-Linux)进行远程调试…...

告别console.log式调试:VSCode AI智能变量推演与上下文回溯技术(仅限VSCode 1.89+私有API)

更多请点击: https://intelliparadigm.com 第一章:告别console.log式调试:VSCode AI智能变量推演与上下文回溯技术(仅限VSCode 1.89私有API) VSCode 1.89 引入了基于 Language Server Protocol 扩展的私有调试增强 AP…...

保姆级教程:用LIBERO和Python一步步调试机器人视觉,从相机画面到关节控制

从像素到动作:LIBERO机器人视觉调试实战指南 当你第一次看到机器人通过摄像头"看"到的世界时,那些二维像素阵列背后隐藏着怎样的三维空间信息?如何让这些抽象的数字转化为精确的机械臂运动?本文将带你像侦探破案一样&am…...

web权限提升与转移学习笔记

参考小迪安全高端No.1环境准备本地搭建demo访问后台admin/123456Tmall-后台权限->Web权限(提升)登录完成后由于java开发的网站利用哥斯拉生成后门来到文件上传功能上传1.jpg改包1.jsp发送这里我一开始用localhost抓不到包。换成本机真实IP立即解决&am…...

暗黑3终极按键助手:5分钟打造你的专属自动化战斗系统

暗黑3终极按键助手:5分钟打造你的专属自动化战斗系统 【免费下载链接】D3keyHelper D3KeyHelper是一个有图形界面,可自定义配置的暗黑3鼠标宏工具。 项目地址: https://gitcode.com/gh_mirrors/d3/D3keyHelper 还在为暗黑3中繁琐的技能连点而手指…...

告别数据抖动!手把手教你配置SGM58200 ADC的50/60Hz工频抗干扰采样(附STM32 I2C代码)

工业级ADC抗干扰实战:SGM58200精准抑制50/60Hz工频噪声的配置指南 在工业测量和传感器信号采集中,工频干扰就像一位不请自来的"噪音制造者"。当你的精密仪器读数出现周期性波动,或是数据采集结果出现难以解释的抖动时,很…...

大语言模型评估指标全解析与应用实践

1. 大语言模型评估指标入门指南 在自然语言处理领域,大语言模型(LLM)的评估一直是个令人头疼的问题。不同于传统机器学习任务有明确的准确率、召回率等指标,LLM的评估需要考虑语言质量、连贯性、事实准确性、创造性等多个维度。我曾在三个不同的LLM项目中…...

如何一键完成Windows和Office智能激活:KMS_VL_ALL_AIO完整指南

如何一键完成Windows和Office智能激活:KMS_VL_ALL_AIO完整指南 【免费下载链接】KMS_VL_ALL_AIO Smart Activation Script 项目地址: https://gitcode.com/gh_mirrors/km/KMS_VL_ALL_AIO 还在为Windows系统激活烦恼吗?Office软件突然变成只读模式…...

C++26反射元编程架构设计图首次公开(ISO/IEC JTC1 SC22 WG21内部评审版):含3层抽象边界定义与21个编译期约束断言

更多请点击: https://intelliparadigm.com 第一章:C26反射元编程架构设计图概览 C26 正式引入标准化的编译时反射(std::reflect)核心设施,标志着元编程范式从模板元编程(TMP)和 constexpr 编程…...