0

0

Java面试之分布式锁的常见实现方案对比

畫卷琴夢

畫卷琴夢

发布时间:2026-01-08 08:14:17

|

503人浏览过

|

来源于php中文网

原创

Redis SET 命令的 NX 和 EX 选项合并操作可原子实现加锁与设过期时间,避免 SETNX 后 EXPIRE 失败导致死锁;ZooKeeper 临时顺序节点依赖 ZAB 协议强一致性和 Session 自动清理,规避假死问题;数据库乐观锁本质是并发控制,非真正分布式锁,适用场景有限。

java面试之分布式锁的常见实现方案对比

Redis SETNX + Lua 脚本为什么不能直接用 SETNX

单独调用 SETNX 只能保证“存在性判断+设值”原子性,但无法同时设置过期时间。如果先 SETNX 成功、再 EXPIRE 失败(如网络中断或进程崩溃),就会留下永不过期的死锁锁。

正确做法是用 SET 命令的 NXEX 选项合并操作:

SET lock:order:123 "8a9b" NX EX 30

其中 "8a9b" 是客户端唯一标识(用于后续校验和释放),30 是过期秒数。这个命令在 Redis 2.6.12+ 支持,天然原子。

常见踩坑点:

立即学习Java免费学习笔记(深入)”;

  • 误用老版本 Redis(NX 和 EX 不被识别,返回错误或静默失败
  • 没传客户端唯一值,导致任意客户端都能删锁(DEL 无校验)
  • 过期时间设太短,业务未执行完锁就自动释放,引发并发冲突

Redisson 的 RLock 是怎么避免误删锁的?

RLock 在加锁时自动生成唯一 threadId + UUID 作为锁 value,并用 Lua 脚本封装“判断 value 相等再 DEL”逻辑,确保只有加锁者能解锁。

它还支持自动续期(watchdog):默认 30 秒锁过期,但只要客户端还活着,每 10 秒会自动延长锁有效期。这解决了“业务耗时 > 锁 TTL”的经典问题。

但要注意:

Live PPT
Live PPT

一款AI智能化生成演示内容的在线工具。只需输入一句话、粘贴一段内容、或者导入文件,AI生成高质量PPT。

下载
  • watchdog 依赖 Redisson 客户端心跳,若 JVM Full GC 时间过长(> 心跳间隔),可能导致续期失败而提前释放锁
  • 非 Redisson 客户端(如 Jedis)加的锁,RLock.unlock() 无法识别 value 格式,会直接报错或误删
  • RedLock 模式(多 Redis 实例)在多数场景下收益远低于复杂度,官方已不推荐,除非你有跨机房容灾强需求

ZooKeeper 的临时顺序节点方案为何适合强一致性场景?

ZooKeeper 依靠 ZAB 协议保证写入强一致,锁释放不依赖超时,而是靠 Session 断连后自动删除临时节点,彻底规避“假死导致锁残留”问题。

典型流程是:客户端创建临时顺序节点(如 /lock/worker-000000001),再检查自己是否为最小序号节点;不是则监听前一个节点的删除事件。

优势与代价并存:

  • 没有过期时间概念,锁不会因网络抖动意外失效
  • ZK 集群性能瓶颈明显,高并发抢锁易成瓶颈(尤其 watch 通知风暴)
  • 客户端需处理连接丢失、session 过期、重复 watch 等大量状态逻辑,比 Redis 方案重得多
  • 节点路径必须带业务前缀隔离,否则不同服务共用同一 /lock 节点会互相干扰

数据库乐观锁在分布式环境下的适用边界在哪?

基于数据库的 UPDATE ... WHERE version = ? 是典型的乐观锁,但它本质不是“分布式锁”,而是“并发更新控制”。它不阻塞请求,只拒绝冲突写入。

真正用作分布式锁时(如插入唯一键记录),性能和可靠性都受限:

  • 唯一索引冲突抛异常(SQLIntegrityConstraintViolationException)才能感知锁失败,链路不直观
  • MySQL 的 INSERT ... ON DUPLICATE KEY UPDATE 或 PostgreSQL 的 INSERT ... ON CONFLICT DO NOTHING 可用,但吞吐量远低于 Redis
  • DB 连接池打满、主从延迟、事务超时都会放大锁失败率,不适合低延迟敏感型业务
  • 无法实现可重入、等待队列、公平性等高级语义,仅适合简单“有/无”抢占场景

真正需要分布式锁时,优先选 Redis(简单高效)或 ZooKeeper(强一致刚需);用数据库,往往是已有系统改造成本低的权衡,不是技术首选。

相关专题

更多
java
java

Java是一个通用术语,用于表示Java软件及其组件,包括“Java运行时环境 (JRE)”、“Java虚拟机 (JVM)”以及“插件”。php中文网还为大家带了Java相关下载资源、相关课程以及相关文章等内容,供大家免费下载使用。

829

2023.06.15

java正则表达式语法
java正则表达式语法

java正则表达式语法是一种模式匹配工具,它非常有用,可以在处理文本和字符串时快速地查找、替换、验证和提取特定的模式和数据。本专题提供java正则表达式语法的相关文章、下载和专题,供大家免费下载体验。

733

2023.07.05

java自学难吗
java自学难吗

Java自学并不难。Java语言相对于其他一些编程语言而言,有着较为简洁和易读的语法,本专题为大家提供java自学难吗相关的文章,大家可以免费体验。

733

2023.07.31

java配置jdk环境变量
java配置jdk环境变量

Java是一种广泛使用的高级编程语言,用于开发各种类型的应用程序。为了能够在计算机上正确运行和编译Java代码,需要正确配置Java Development Kit(JDK)环境变量。php中文网给大家带来了相关的教程以及文章,欢迎大家前来阅读学习。

396

2023.08.01

java保留两位小数
java保留两位小数

Java是一种广泛应用于编程领域的高级编程语言。在Java中,保留两位小数是指在进行数值计算或输出时,限制小数部分只有两位有效数字,并将多余的位数进行四舍五入或截取。php中文网给大家带来了相关的教程以及文章,欢迎大家前来阅读学习。

398

2023.08.02

java基本数据类型
java基本数据类型

java基本数据类型有:1、byte;2、short;3、int;4、long;5、float;6、double;7、char;8、boolean。本专题为大家提供java基本数据类型的相关的文章、下载、课程内容,供大家免费下载体验。

446

2023.08.02

java有什么用
java有什么用

java可以开发应用程序、移动应用、Web应用、企业级应用、嵌入式系统等方面。本专题为大家提供java有什么用的相关的文章、下载、课程内容,供大家免费下载体验。

430

2023.08.02

java在线网站
java在线网站

Java在线网站是指提供Java编程学习、实践和交流平台的网络服务。近年来,随着Java语言在软件开发领域的广泛应用,越来越多的人对Java编程感兴趣,并希望能够通过在线网站来学习和提高自己的Java编程技能。php中文网给大家带来了相关的视频、教程以及文章,欢迎大家前来学习阅读和下载。

16925

2023.08.03

Golang 分布式缓存与高可用架构
Golang 分布式缓存与高可用架构

本专题系统讲解 Golang 在分布式缓存与高可用系统中的应用,涵盖缓存设计原理、Redis/Etcd集成、数据一致性与过期策略、分布式锁、缓存穿透/雪崩/击穿解决方案,以及高可用架构设计。通过实战案例,帮助开发者掌握 如何使用 Go 构建稳定、高性能的分布式缓存系统,提升大型系统的响应速度与可靠性。

6

2026.01.09

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
MySQL 教程
MySQL 教程

共48课时 | 1.7万人学习

MySQL 初学入门(mosh老师)
MySQL 初学入门(mosh老师)

共3课时 | 0.3万人学习

简单聊聊mysql8与网络通信
简单聊聊mysql8与网络通信

共1课时 | 785人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号