【知识摘要】一文带你了解什么是RedLock。

1、什么是RedLock

红锁(RedLock)是一种分布式锁算法,由 Redis 的作者 Salvatore Sanfilippo(也称为 Antirez)设计,用于在分布式系统中实现可靠的锁机制。它的设计解决了单一 Redis 实例作为分布式锁可能出现的单点故障问题。

红锁(RedLock)是一种分布式锁算法,由 Redis 的作者 Salvatore Sanfilippo(也称为 Antirez)设计,用于在分布式系统中实现可靠的锁机制。它的设计解决了单一 Redis 实例作为分布式锁可能出现的单点故障问题。

实现原理:

  1. 多节点加锁: RedLock 不在单个 Redis 实例上加锁,而是在多个独立的 Redis 实例上同时尝试获取锁。通常建议使用奇数个 Redis 实例(如 5 个),以确保系统具有较好的容错性。
  2. 多数节点同意: 系统只有在获得了大多数 Redis 实例的锁(即 N/2 + 1 个节点,N 为节点总数)之后,才认为成功获取了分布式锁。这样即使部分 Redis 实例发生故障,整体锁服务仍然可用。
  3. 时间同步: 为防止客户端在持有锁的过程中发生故障而导致锁无法释放,RedLock 会在获取锁时设置一个超时时间。如果客户端在锁超时之前未能完成任务并释放锁,其他客户端可以在锁超时后重新尝试获取。
  4. 锁释放: 释放锁时,客户端需要向所有 Redis 实例发送释放锁的命令,以确保所有实例上的锁都被清除。

工作流程:

  1. 客户端尝试顺序地向所有 Redis 实例发送加锁命令。
  2. 对于每个实例,客户端尝试在指定的超时时间内获取锁。
  3. 客户端计算已经成功加锁的实例数量,如果达到多数(N/2 + 1),则认为客户端成功获取了分布式锁。
  4. 如果获取锁失败,客户端需要向所有实例发送释放锁的命令,以避免留下未释放的锁。

在 Java 中的应用:
在 Java 中,可以使用 Redisson 框架来实现 RedLock。Redisson 提供了 RedissonMultiLock 类,它可以同时管理多个锁,并保证操作的原子性。

以下是 Redisson 中 RedLock 的简单使用示例:

RedissonClient redisson = // 初始化 Redisson 客户端
RLock lock1 = redisson.getLock("lock1");
RLock lock2 = redisson.getLock("lock2");
RLock lock3 = redisson.getLock("lock3");RedissonMultiLock multiLock = new RedissonMultiLock(lock1, lock2, lock3);
try {if (multiLock.tryLock()) {// 成功获取锁,执行业务逻辑} else {// 获取锁失败}
} finally {multiLock.unlock(); // 释放锁
}

RedissonRedLock 实际上是基于 RedissonMultiLock 实现的,从继承关系可以看出这一点。

通过以上机制,RedLock 在分布式环境下提供了一种较为可靠的锁方案,能够应对部分节点故障,并保持锁服务的可用性和安全性。

2、RedLock主要特性

RedLock 具备以下主要特性:

  • 互斥性:在任何时间,只有一个客户端可以获得锁,确保了资源的互斥访问。
  • 避免死锁:通过为锁设置一个较短的过期时间,即使客户端在获得锁后由于网络故障等原因未能按时释放锁,锁也会因为过期而自动释放,避免了死锁的发生。
  • 容错性:即使一部分 Redis 节点宕机,只要大多数节点(即过半数以上的节点)仍在线,RedLock 算法就能继续提供服务,并确保锁的正确性。

3、存在问题

RedLock 由于其设计原理和实现上的复杂性,存在一些问题和争议。您提到的性能问题和并发安全性问题是其中比较关键的。

性能问题

由于 RedLock 需要在多个节点间进行交互,网络延迟和节点超时确实可能影响加锁的性能。特别是在节点数量较多或网络状况不佳的情况下,这种影响会更加明显。

并发安全性问题

您描述的场景是一个典型的并发问题,即客户端在持有锁的过程中发生长时间停顿(例如 JVM 的 STW),导致锁实际上已经失效,但客户端由于停顿结束后仍然认为持有锁。

4、RedLock 被废弃

由于这些问题,RedLock 在一些场景下可能不是最佳选择。Redisson 官方已经废弃了 RedLock,这也反映了分布式系统设计中的一些挑战。

替代方案

对于分布式锁的需求,以下是一些替代方案:

基于单 Redis 节点的分布式锁:
如果对性能要求较高,且能够接受单点故障的风险,可以使用基于单个 Redis 实例的分布式锁。
可以使用 Redisson 提供的 RLock 或 FairLock,并通过主从复制或哨兵模式来提高可用性。

基于 ZooKeeper 的分布式锁:
ZooKeeper 提供了原生的分布式锁实现,通过其临时节点和顺序节点的特性,可以创建可靠的分布式锁。
ZooKeeper 的分布式锁比较重,但在一致性方面表现较好。

基于 etcd 的分布式锁:
etcd 是另一个分布式键值存储系统,它也可以用来实现分布式锁。
etcd 的 watch 机制可以用来监听锁状态,从而实现锁的自动释放和重试逻辑。

基于数据库的分布式锁:
通过数据库的唯一约束或乐观锁来实现分布式锁。
这种方法通常依赖于数据库的事务和锁机制。

使用集群化 Redis 或 Redis Module:
使用 Redis 集群,结合 Redisson,可以在提高可用性的同时减少单点故障的风险。
Redis Modules(如 RediSearch、RedisBloom)也可以用于实现分布式锁。

5、总结

RedLock是一种分布式锁算法,由Redis的作者Salvatore Sanfilippo设计,用于在分布式系统中实现可靠的锁机制。其核心思想在多个独立的Redis实例上同时获取锁,只有当大多数Redis实例加锁成功时,才认为成功获取了分布式锁

RedLock通过多节点加锁、多数节点同意、时间同步和锁释放机制,提高了分布式锁的可用性和安全性。然而,RedLock也存在一些问题,如性能问题和并发安全性问题,并且由于这些问题,Redisson中已经废弃了RedLock。

对于分布式锁的需求,可以考虑使用基于单Redis节点的分布式锁、基于ZooKeeper的分布式锁、基于etcd的分布式锁、基于数据库的分布式锁或使用集群化Redis或Redis Module等替代方案。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/712890.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

【Django】执行查询—跨关系查询中的跨多值关联问题

跨多值查询 跨越 ManyToManyField 或反查 ForeignKey (例如从 Blog 到 Entry )时,对多个属性进行过滤会产生这样的问题:是否要求每个属性都在同一个相关对象中重合。 filter() 先看filter(),通过一个例子看&#xf…

打造无缝滚动体验:JavaScript中的scrollIntoView()方法实战指南

在现代Web开发中,提升用户体验是至关重要的。通过JavaScript的scrollIntoView()方法,我们可以为用户创造出流畅而令人愉悦的滚动体验。本文将深入研究scrollIntoView()的强大功能,并结合实例演示如何在项目中巧妙应用,以打造出无缝…

缓存穿透解决方案之布隆过滤器

布隆过滤器可以快速判断数据是否存在,避免从数据库中查询数据是否存在,减轻数据库的压力 布隆过滤器是由一个初值为0的bit数组和N个哈希函数,可以用来快速的判断某个数据是否存在 当我们想要标记某个数据是否存在时,布隆过滤器会…

Java底层自学大纲_高可用篇

高可用专题_自学大纲所属类别学习主题建议课时(h) A 容器化技术001 Docker架构设计原理2.5 A 容器化技术002 Docker部署springboot项目2.5 A 容器化技术003 基于Docker-Compose部署微服务项目2.5 B Nginx实现高可用004 Nginx反向代理&负载均衡&a…

LabVIEW眼结膜微血管采集管理系统

LabVIEW眼结膜微血管采集管理系统 开发一套基于LabVIEW的全自动眼结膜微血管采集管理系统,以提高眼结膜微血管临床研究的效率。系统集成了自动化图像采集、图像质量优化和规范化数据管理等功能,有效缩短了图像采集时间,提高了图像质量&#…

idea 多模块A模块调用了B模块的Jar包,而非本地源码

1,问题描述 对于多模块的互相调用,比如模块A,模块B,模块C, 这在本地都是可以编辑进行开发的源码, 按理说是模块A可以直接点进模块B的本地源码, 但是不知道什么原因,导致模块A点进…

C++小记 - 二叉树

文章目录 二叉树一、二叉树理论基础篇二叉树的种类满二叉树完全二叉树二叉搜索树平衡二叉搜索树 二叉树的存储方式链式存储:顺序存储:遍历规则:构造实现: 二叉树的遍历方式二叉树的定义 二、二叉树的递归遍历递归算法的三个要素:递…

vue+element UI中给指定日期添加标记

1.日期控件中添加:picker-options属性&#xff0c;即:picker-options“myPickerOptions” <el-date-picker:class"item.scds !null ?xtsjBlue:xtsjRed"v-model"item.date"value-format"yyyy-MM-dd"type"date":picker-options"…

Python中的heapq模块

Python中的heapq模块 文章目录 Python中的heapq模块1.heapq的方法2.使用heapq创建堆3.使用heapq实现堆排序4.获取堆中的前n个最大值或最小值Reference heapq模块实现了堆队列的算法&#xff0c;即优先队列算法。heapq其实是实现了一种小顶堆&#xff0c;所以使用pop()方法返回的…

如何进行弱网测试?

&#x1f345; 视频学习&#xff1a;文末有免费的配套视频可观看 &#x1f345; 点击文末小卡片&#xff0c;免费获取软件测试全套资料&#xff0c;资料在手&#xff0c;涨薪更快 如今这个高度互联的时代里&#xff0c;网络环境对于应用程序的影响越来越重要。 而弱网测试就是…

leetcode--接雨水(双指针法,动态规划,单调栈)

目录 方法一&#xff1a;双指针法 方法二&#xff1a;动态规划 方法三&#xff1a;单调栈 42. 接雨水 - 力扣&#xff08;LeetCode&#xff09; 黑色的是柱子&#xff0c;蓝色的是雨水&#xff0c;我们先来观察一下雨水的分布情况: 雨水落在凹槽之间&#xff0c;在一个凹槽的…

使用js写一个登录验证码效果

面试题 登录页面获取验证码的功能&#xff0c;用户点击获取验证码按钮(id”btn1”)&#xff0c;按文字变为“(N)后获取验证码”&#xff0c;N为倒计对秒数&#xff0c;从 60 开始&#xff0c;每秒减一&#xff0c;减到 0的时候&#xff0c;按钮文字变为“获取验证码”&#xff…

Beans模块之工厂模块Aware

博主介绍:✌全网粉丝5W+,全栈开发工程师,从事多年软件开发,在大厂呆过。持有软件中级、六级等证书。可提供微服务项目搭建与毕业项目实战,博主也曾写过优秀论文,查重率极低,在这方面有丰富的经验✌ 博主作品:《Java项目案例》主要基于SpringBoot+MyBatis/MyBatis-plus+…

【JavaWeb】

Javaweb 数据库相关概念MySQL数据库MySQL数据模型SQLDDL--操作数据库图形化客户端工具DML--操作数据DQL数据库约束 数据库设计多表查询事务 数据库相关概念 数据库 存储数据的仓库&#xff0c;数据是有组织的进行存储 英文&#xff1a;DataBase&#xff0c;简称DB 数据库管理系…

单元测试数据库回滚问题

问题现象&#xff1a; 在进行单元测试时&#xff0c;测试执行成功&#xff0c;可是数据库中的数据没变 问题解决&#xff1a;单元测试自动回滚&#xff0c;需要加上注解Rollback(false) https://zhhll.icu/2020/javaweb/问题/1.单元测试数据问题/ 本文由 mdnice 多平台发布

机器学习-3

文章目录 前言训练验证测试评估评估方法交叉验证法自助法评估指标 练习题 前言 本篇介绍机器学习中的训练、验证、测试与评估的相关概念。 训练 从数据中学得模型的过程称为“学习”(learning)或“训练”(training),这个过程通过执行某个学习算法来完成.训练过程中使用的数据…

Android T 远程动画显示流程其三——桌面侧动画启动到系统侧结束流程

前言 接着前文分析Android T 远程动画显示流程其二 我们通过IRemoteAnimationRunner跨进程通信从系统进程来到了桌面进程&#xff0c;这里是真正动画播放的逻辑。 之后又通过IRemoteAnimationFinishedCallback跨进程通信回到系统进程&#xff0c;处理动画结束时的逻辑。 进入…

使用maven项目引入jQuery

最近在自学 springBoot &#xff0c;期间准备搞一个前后端不分离的东西&#xff0c;于是需要在 maven 中引入jQuery 依赖&#xff0c;网上百度了很多&#xff0c;这里来做一个总结。 1、pom.xml 导入依赖 打开我们项目的 pom.xml 文件&#xff0c;输入以下坐标。这里我使用的是…

FPGA-学会使用vivado中的存储器资源ROM(IP核)

问题&#xff1a; 某芯片,有500个寄存器,需要在上电的时候由FPGA向这些寄存器中写入初始值,初始值已经通过相应的文档给出了具体值,这些值都是已知的。 分析关键点&#xff1a; 数据量比较多&#xff08;Verilog代码&#xff0c;通过case语句、always语句这种查找表的方式,数…

Linux——匿名管道

Linux——匿名管道 什么是管道匿名管道的底层原理观察匿名管道现象读写端的几种情况写端慢&#xff0c;读端快写端快&#xff0c;读端慢 管道的大小写端关闭&#xff0c;读端一直读写端一直写&#xff0c;读端关闭 我们之前一直用的是vim来编写代码&#xff0c;现在有了vscode这…