0

0

redis如何实现分布式锁 redis分布式锁的5种实现方式对比

冰火之心

冰火之心

发布时间:2025-07-20 11:12:02

|

762人浏览过

|

来源于php中文网

原创

redis分布式锁的常见坑包括锁的误删除和过期时间设置不合理。针对锁的误删除,解决方法是将锁的value设为客户端唯一标识,并通过lua脚本原子性判断后再释放锁;针对过期时间问题,可采用“看门狗”机制自动续期。此外,常见的5种实现方式各有优劣:1.setnx+expire非原子操作易导致死锁,仅适用于学习;2.setnx+lua脚本解决原子性和误删问题,但需维护脚本;3.set命令扩展参数(nx/ex)推荐使用,简洁高效且原子性强;4.redlock算法提高可用性但复杂度高,适用于高要求场景;5.redisson框架封装完善、功能丰富,适合java项目使用。选择实现方式应根据业务需求权衡可靠性、可用性和系统复杂度。

redis如何实现分布式锁 redis分布式锁的5种实现方式对比

Redis 实现分布式锁,核心在于利用其单线程特性和原子操作,确保在分布式环境下只有一个客户端能获得锁。对比多种实现方式,我们需要关注性能、可靠性、复杂度和容错性等因素。

Redis 实现分布式锁,通常会使用 SETNX(SET if Not eXists)命令,它只有在 key 不存在时才设置 key 的值。拿到锁的客户端可以执行业务逻辑,完成后释放锁(通常使用 DEL 命令)。但简单的 SETNX + DEL 存在很多问题,需要进一步完善。

Redis 分布式锁有哪些常见的坑?如何避免?

最常见的坑就是锁的误删除。客户端 A 获得了锁,执行业务逻辑的时间超过了锁的过期时间,Redis 自动释放了锁。此时,客户端 B 获得了锁。随后,客户端 A 完成了业务逻辑,执行 DEL 命令,却把客户端 B 的锁给释放了。

为了避免这种情况,可以在设置锁的时候,将 value 设置为客户端的唯一标识(比如 UUID)。释放锁的时候,先判断锁的 value 是否是自己的标识,如果是才执行 DEL 命令。这个判断和 DEL 命令需要是原子性的,可以使用 Lua 脚本来实现。

另外一个坑是锁的过期时间设置不合理。如果过期时间设置太短,可能导致锁提前释放,引发并发问题;如果过期时间设置太长,可能导致锁无法及时释放,影响系统的可用性。

解决这个问题,可以采用“看门狗”机制。客户端在获得锁之后,启动一个后台线程,定期检查锁是否快要过期。如果快要过期,就自动续期。Redisson 框架就实现了这种机制。

Redis 分布式锁的 5 种实现方式对比

  1. SETNX + EXPIRE (最基础的实现)

    千问APP
    千问APP

    阿里最强大模型官方AI助手

    下载
    • 实现: 使用 SETNX 获取锁,EXPIRE 设置过期时间。
    • 优点: 简单易懂。
    • 缺点: 非原子操作,SETNX 成功后,EXPIRE 失败会导致死锁。
    • 适用场景: 学习和演示,生产环境不推荐。
  2. SETNX + Lua 脚本

    • 实现: 使用 SETNX 获取锁,通过 Lua 脚本原子性地设置过期时间。释放锁时,也通过 Lua 脚本判断锁的持有者是否是自己,防止误删。
    • 优点: 解决了 SETNX + EXPIRE 的非原子性问题,以及锁的误删除问题。
    • 缺点: 需要编写和维护 Lua 脚本。
    • 适用场景: 对可靠性有一定要求的场景。
  3. SET 命令扩展参数 (SET key value [EX seconds] [PX milliseconds] [NX|XX])

    • 实现: Redis 2.6.12 之后,SET 命令增加了 NX (不存在才设置), XX (存在才设置), EX (秒级过期), PX (毫秒级过期) 等参数,可以原子性地完成 SETNX + EXPIRE 的功能。
    • 优点: 原子性操作,代码简洁。
    • 缺点: 需要 Redis 版本支持。
    • 适用场景: 推荐使用,简单高效。
    SET lock_key unique_id NX EX 10
  4. Redlock 算法

    • 实现: 尝试从 N 个独立的 Redis 实例获取锁,只有超过半数成功才认为获取锁成功。释放锁时,需要释放所有 Redis 实例上的锁。
    • 优点: 提高了锁的可用性,即使部分 Redis 实例宕机,锁仍然可以正常工作。
    • 缺点: 实现复杂,性能较低,需要维护多个 Redis 实例,存在争议。
    • 适用场景: 对可用性要求极高的场景,例如金融支付。
  5. Redisson 框架

    • 实现: Redisson 是一个 Redis 的 Java 客户端,提供了分布式锁的实现,包括可重入锁、公平锁、联锁、红锁等。它实现了看门狗机制,自动续期,解决了锁的过期时间问题。
    • 优点: 封装了分布式锁的各种细节,使用简单,功能强大。
    • 缺点: 引入了额外的依赖,增加了系统的复杂度。
    • 适用场景: Java 项目,需要使用分布式锁,但不想自己实现。

如何选择合适的 Redis 分布式锁实现方式?

选择哪种实现方式,需要根据具体的业务场景和需求来决定。

  • 如果只是学习和演示,或者对可靠性要求不高,可以使用 SETNX + EXPIRE
  • 如果对可靠性有一定要求,可以使用 SETNX + Lua 脚本SET 命令扩展参数
  • 如果对可用性要求极高,可以使用 Redlock 算法。
  • 如果是 Java 项目,并且不想自己实现分布式锁,可以使用 Redisson 框架。

需要注意的是,即使使用了分布式锁,也无法保证绝对的安全性。在设计系统时,还需要考虑其他的并发控制手段,例如乐观锁、悲观锁等。同时,要做好监控和报警,及时发现和处理问题。另外,不要过度依赖分布式锁,尽量减少锁的竞争,提高系统的性能。

相关专题

更多
java
java

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

837

2023.06.15

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

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

741

2023.07.05

java自学难吗
java自学难吗

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

736

2023.07.31

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

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

397

2023.08.01

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

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

399

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中文网给大家带来了相关的视频、教程以及文章,欢迎大家前来学习阅读和下载。

16926

2023.08.03

PHP WebSocket 实时通信开发
PHP WebSocket 实时通信开发

本专题系统讲解 PHP 在实时通信与长连接场景中的应用实践,涵盖 WebSocket 协议原理、服务端连接管理、消息推送机制、心跳检测、断线重连以及与前端的实时交互实现。通过聊天系统、实时通知等案例,帮助开发者掌握 使用 PHP 构建实时通信与推送服务的完整开发流程,适用于即时消息与高互动性应用场景。

3

2026.01.19

热门下载

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

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
进程与SOCKET
进程与SOCKET

共6课时 | 0.3万人学习

Redis+MySQL数据库面试教程
Redis+MySQL数据库面试教程

共72课时 | 6.4万人学习

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

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