news 2026/4/26 5:06:15

分布式锁的七种写法:从 SetNX 到 Redisson 看门狗 (WatchDog),谁才是最终答案?

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
分布式锁的七种写法:从 SetNX 到 Redisson 看门狗 (WatchDog),谁才是最终答案?

🔒 前言:面试官的陷阱

“请手写一个 Redis 分布式锁。”
如果你直接写了redis.setnx(key, 1),面试官可能会让你直接回家等通知。

为什么?

  • 如果服务器宕机,锁没释放怎么办?(死锁)
  • 如果你加了过期时间,但业务执行时间超长怎么办?(锁失效)
  • 如果你释放了别人的锁怎么办?(并发安全)

今天,我们像剥洋葱一样,从青铜到王者,拆解分布式锁的7 种进化形态


👶 第一阶段:青铜时代 (漏洞百出)

写法一:裸奔的setnx
// 1. 抢锁if(redis.setnx(lockKey,1)==1){try{// 业务逻辑...}finally{// 2. 释放锁redis.del(lockKey);}}
  • 致命死穴:如果业务逻辑执行到一半,服务器宕机了/重启了finally没执行,锁永远删不掉。造成永久死锁!
写法二:setnx+expire
if(redis.setnx(lockKey,1)==1){redis.expire(lockKey,30);// 补一个过期时间try{...}finally{redis.del(lockKey);}}
  • 致命死穴setnxexpire是两条命令,不是原子的。如果刚 setnx 完,网线被拔了,expire 没执行,依然死锁!

🥈 第二阶段:白银时代 (原子性解决)

写法三:SET NX PX(Redis 2.6.12+)

Redis 官方终于看不下去了,把设置值和过期时间合并成了一条命令。

// 原子命令:SET lockKey requestId NX PX 30000Stringresult=jedis.set(lockKey,requestId,"NX","PX",30000);if("OK".equals(result)){try{...}finally{redis.del(lockKey);}}
  • 潜在问题:解决了死锁,但还有一个大坑——误删锁
    1. A 拿到锁,过期时间 30s。
    2. A 业务卡顿,跑了 40s。此时锁自动过期,B 拿到了锁。
    3. A 跑完了,执行finally里的del把 B 的锁给删了!
    4. C 趁虚而入,B 和 C 同时在跑,线程不安全。
写法四:UUID + Lua 脚本 (校验身份)

为了防止删错锁,我们在 Value 里存一个 UUID (Client ID)。删除前判断一下:这是不是我的锁?

-- Lua 脚本保证原子性ifredis.call('get',KEYS[1])==ARGV[1]thenreturnredis.call('del',KEYS[1])elsereturn0end
  • 依然痛点:虽然不删别人的锁了,但 A 的业务还没跑完锁就过期了,A 依然是在“裸奔”(并发执行)。我们需要**“锁续期”**。

🥇 第三阶段:黄金时代 (Redisson 看门狗)

写法五:手动写守护线程 (太累)

你自己写一个 Timer,每隔 10秒 检查一下,如果 A 还在跑,就给锁续命。
但是,写好这个多线程逻辑极其复杂,很容易由“解决死锁”变成“制造死锁”。

写法六:Redisson (工业级标准)

Redisson是 Redis 官方推荐的 Java 客户端,它内置了一个神器——看门狗 (WatchDog)

使用代码(极其优雅):

RLocklock=redisson.getLock("myLock");try{// 1. 加锁 (默认 30s 过期,自动启动看门狗)lock.lock();// 2. 业务逻辑 (哪怕跑 1 小时,锁也不会断)Thread.sleep(60*60*1000);}finally{// 3. 解锁 (自动停止看门狗)lock.unlock();}

看门狗原理图解:

看门狗续期机制
1. 尝试加锁 (SET NX PX)
2. 加锁成功
3. 启动 WatchDog (后台线程)
4. 每隔 10s (lockWatchdogTimeout/3)
是 (业务未完)
6. 业务结束 unlock()
7. 停止 WatchDog
检测锁还在吗?
看门狗任务
5. 重置过期时间为 30s
Redis 服务端
客户端线程 A

WatchDog 核心逻辑:

  1. Redisson 只要加锁成功,就会启动一个后台定时任务(TimeTask)。
  2. 默认每隔10秒(默认过期时间 30s 的 1/3) 检查一次。
  3. 如果当前线程还持有锁,就通过 Lua 脚本把 Redis 里的过期时间重置为 30s。
  4. 如果服务器宕机,看门狗线程也挂了,没人续期,Redis 里的锁 30s 后自动失效,不会死锁。完美!

👑 第四阶段:王者时代 (RedLock 红锁)

写法七:RedLock (解决主从一致性)

场景

  1. A 在 Redis Master 拿到了锁。
  2. Master 还没来得及把数据同步给 Slave,Master 挂了。
  3. Slave 升级为 New Master。
  4. B 在 New Master 也可以拿到锁。
    结果:A 和 B 同时持锁。

为了解决这个极端问题(虽然概率极低),Redisson 实现了RedLock算法。

  • 原理:搞 5 个独立的 Redis 节点(不是集群,没有主从)。
  • 规则:客户端同时向这 5 个节点申请锁,只要N/2 + 1 (即 3 个)节点加锁成功,就认为获取锁成功。
RLocklock1=redisson1.getLock("lock");RLocklock2=redisson2.getLock("lock");RLocklock3=redisson3.getLock("lock");RedissonRedLocklock=newRedissonRedLock(lock1,lock2,lock3);lock.lock();

(注:RedLock 性能较差,且存在时钟跳变问题,业界争议较大,一般业务场景不推荐使用。)


📝 总结:谁才是最终答案?

在 99% 的业务场景下(电商秒杀、库存扣减、定时任务不重跑),Redisson 的默认锁(写法六)就是最终答案

它完美解决了:

  1. 死锁(宕机自动过期)
  2. 原子性(Lua 脚本)
  3. 误删(UUID 校验)
  4. 业务超时(WatchDog 自动续期)

除非你是在做“金融级核心转账”,否则不要去碰 RedLock,过度设计是万恶之源。


版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/4/26 9:35:44

GHOST误操作致分区消失?一招教你找回丢失数据

在数据备份与恢复的领域里,GHOST作为一款曾经广泛使用的系统备份与还原工具,为众多用户带来了便利。然而,不当的操作或误选选项,有时会导致意想不到的后果——比如,原本多分区存储的数据,在GHOST操作后&…

作者头像 李华
网站建设 2026/4/21 6:15:04

前端剪贴板功能实现方案:从零到一的完整技术指南

前端剪贴板功能实现方案:从零到一的完整技术指南 【免费下载链接】clipboard.js :scissors: Modern copy to clipboard. No Flash. Just 3kb gzipped :clipboard: 项目地址: https://gitcode.com/gh_mirrors/cl/clipboard.js 前端剪贴板功能在现代Web应用中扮…

作者头像 李华
网站建设 2026/4/23 3:41:03

亲测!专业机油批发的实践与成果

亲测!专业机油批发的实践与成果引言在汽车后市场中,机油批发业务具有广阔的发展前景。四川小江军作为一家专注于机油领域的品牌,在专业机油批发方面有着丰富的实践经验和显著的成果。本文将结合亲测体验,探讨专业机油批发的实践要…

作者头像 李华
网站建设 2026/4/20 13:13:45

如何用字节跳动开源AI助手让工作效率翻倍?

如何用字节跳动开源AI助手让工作效率翻倍? 【免费下载链接】UI-TARS-2B-SFT 项目地址: https://ai.gitcode.com/hf_mirrors/ByteDance-Seed/UI-TARS-2B-SFT 还在为重复的电脑操作烦恼吗?每天花费大量时间在Excel数据处理、邮件整理、文件归档上&…

作者头像 李华
网站建设 2026/4/25 1:59:53

智慧园区系统:技术赋能下的园区管理革新与价值升级

在数字经济加速渗透的今天,智慧园区系统已成为破解传统园区管理瓶颈、激活产业发展动能的核心支撑。它不再是简单的技术叠加,而是集数据整合、智能管控、服务优化于一体的综合性管理生态,通过融合大数据、物联网、云计算、人工智能等前沿技术…

作者头像 李华