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

02_redis分布式锁原理

文章目录

    • 一、redis如何实现分布式锁
      • 1. 使用 SETNX 命令
      • 2. 设置过期时间
      • 3. 释放锁
      • 4. 注意事项
      • 5. 示例代码
    • 二、Java中分布式锁如何设置超时时间
      • 1. Redis分布式锁
      • 2. 基于Zookeeper的分布式锁
      • 3. 基于数据库的分布式锁
      • 注意事项

一、redis如何实现分布式锁

Redis 实现分布式锁是一种常见且高效的方式。以下是关于 Redis 实现分布式锁的一些关键点和步骤:

1. 使用 SETNX 命令

Redis 提供了一个非常方便的命令 SETNX(SET if Not eXists),它可以在指定的 key 不存在时,为 key 设置一个值。这个命令天然地适合用作分布式锁的占位符。

  • 命令格式SETNX key value
  • 返回值:如果 key 不存在,设置 key 的值为 value,返回 1。如果 key 已存在,不做任何操作,返回 0。

2. 设置过期时间

为了防止由于异常情况导致锁无法释放,我们通常需要给锁设置一个过期时间。这可以通过 EXPIRE 命令来实现,或者更优雅地使用 SET 命令的扩展参数。

  • 命令格式EXPIRE key secondsSET key value EX seconds(后者同时设置值和过期时间)

3. 释放锁

释放锁的操作相对简单,只需要删除对应的 key 即可。但是,这里需要注意一个细节:只有当锁的持有者(即设置锁的进程)才能释放锁。因此,在删除锁之前,我们需要检查锁的值是否匹配。

  • 命令格式DEL keyLUA 脚本(为了确保原子性,通常使用 LUA 脚本来检查并删除锁)

4. 注意事项

  • 时钟漂移:在分布式系统中,不同机器的时钟可能存在漂移。因此,设置过期时间时需要考虑这一点,避免由于时钟漂移导致的锁提前释放或过期未释放。
  • 可重入性:如果同一个进程需要多次获取同一个锁,那么锁应该是可重入的。Redis 本身不支持可重入锁,但可以通过在应用层实现一个计数器来解决这个问题。
  • 锁续期:为了避免锁因为操作时间过长而过期,可以在持有锁期间定期续期。这同样需要通过 LUA 脚本来确保原子性。
  • 死锁检测:虽然 Redis 分布式锁本身不会直接导致死锁(因为有过期时间),但应用层仍然需要处理可能出现的死锁情况。例如,可以设置一个监控线程来检测长时间未释放的锁,并进行相应的处理。

5. 示例代码

以下是一个简单的使用 Redis 实现分布式锁的 Java 示例代码(基于 Jedis 库):

import redis.clients.jedis.Jedis;public class RedisDistributedLock {private Jedis jedis;private String lockKey;private String lockValue;private int expireTime;public RedisDistributedLock(Jedis jedis, String lockKey, String lockValue, int expireTime) {this.jedis = jedis;this.lockKey = lockKey;this.lockValue = lockValue;this.expireTime = expireTime;}public boolean tryLock() {String result = jedis.set(lockKey, lockValue, "NX", "EX", expireTime);return "OK".equals(result);}public void unlock() {String script = "if redis.call('get', KEYS[1]) == ARGV[1] then " +"return redis.call('del', KEYS[1]) " +"else " +"return 0 " +"end";jedis.eval(script, 1, lockKey, lockValue);}
}

在这个示例中,tryLock 方法尝试获取锁,如果成功则返回 true,否则返回 falseunlock 方法使用 LUA 脚本来确保只有锁的持有者才能释放锁。

二、Java中分布式锁如何设置超时时间

在Java中实现分布式锁时,设置超时时间是非常重要的,因为它可以防止锁被永远持有(例如,由于客户端崩溃或网络问题导致的锁无法释放)。不同的分布式锁实现方式可能有不同的方法来设置超时时间。以下是一些常见的分布式锁实现及其超时时间设置方法:

1. Redis分布式锁

当使用Redis实现分布式锁时,通常会使用SET命令结合NX(仅当键不存在时设置)和PX(设置键的过期时间,以毫秒为单位)选项来确保原子性和超时控制。

String lockKey = "myLock";
String lockValue = UUID.randomUUID().toString(); // 使用唯一值作为锁的值,以便后续释放锁时验证
int lockExpireTime = 5000; // 锁的超时时间,单位为毫秒Jedis jedis = new Jedis("localhost");
try {String result = jedis.set(lockKey, lockValue, "NX", "PX", lockExpireTime);if ("OK".equals(result)) {// 成功获取锁,执行临界区代码} else {// 获取锁失败,可能已经被其他客户端持有}
} finally {jedis.close();
}

在这个例子中,PX 5000设置了锁的超时时间为5000毫秒(5秒)。

2. 基于Zookeeper的分布式锁

Zookeeper也常用于实现分布式锁。在使用Zookeeper时,可以通过创建临时有序节点来实现锁的功能,并设置节点的超时时间(即会话超时)。

CuratorFramework client = CuratorFrameworkFactory.newClient("localhost:2181", new ExponentialBackoffRetry(1000, 3));
client.start();
try {InterProcessMutex lock = new InterProcessMutex(client, "/my_lock");try {if (lock.acquire(5, TimeUnit.SECONDS)) { // 设置超时时间为5秒try {// 成功获取锁,执行临界区代码} finally {lock.release();}} else {// 获取锁失败}} catch (Exception e) {// 处理异常}
} finally {client.close();
}

在这个例子中,lock.acquire(5, TimeUnit.SECONDS)设置了尝试获取锁的超时时间为5秒。

3. 基于数据库的分布式锁

虽然不太常见,但也可以使用数据库来实现分布式锁。通常,这涉及到在数据库中创建一个锁表,并通过SQL语句来检查、设置和释放锁。超时时间可以通过在数据库中设置锁的过期时间来实现。

注意事项

  • 超时时间的设置:超时时间应该足够长,以确保在正常情况下能够完成任务,但也不能太长,以避免锁被长时间持有而浪费资源。
  • 锁的释放:在释放锁时,需要确保只有持有锁的客户端才能释放锁(例如,通过比较锁的值)。
  • 异常处理:在获取锁和释放锁的过程中,需要妥善处理异常,以避免由于异常导致锁无法被正确释放。
  • 分布式环境的复杂性:在分布式环境中,由于网络延迟、节点故障等原因,实现可靠的分布式锁可能比较复杂。因此,在选择分布式锁实现时,需要仔细考虑其可靠性和性能。

相关文章:

02_redis分布式锁原理

文章目录 一、redis如何实现分布式锁1. 使用 SETNX 命令2. 设置过期时间3. 释放锁4. 注意事项5. 示例代码 二、Java中分布式锁如何设置超时时间1. Redis分布式锁2. 基于Zookeeper的分布式锁3. 基于数据库的分布式锁注意事项 一、redis如何实现分布式锁 Redis 实现分布式锁是一…...

简单血条于小怪攻击模板

创建一个2d正方形(9-Sliced)命名为Player,在Player下面新建一个画布(Canvas)命名为PlayerHealthUI,在画布下面新建一个滑动条(Slider)命名为HealthBar 把PlayerHealthUI脚本挂载到Pl…...

Win11 系统登入时绑定微软邮箱导致用户名欠缺

Win11 系统登入时绑定微软邮箱导致用户名欠缺 解决思路 -> 解绑当前微软邮箱和用户名 -> 断网离线建立本地账户 -> 设置本地账户为Admin权限 -> 注销当前账户,登入新建的用户 -> 联网绑定微软邮箱 -> 删除旧的用户命令步骤 管理员权限打开…...

代码随想录算法训练营第四十六四十七天

卡码网题目: 110. 字符串接龙105. 有向图的完全联通106. 岛屿的周长107. 寻找存在的路径 其他: 今日总结 往期打卡 110. 字符串接龙 跳转: 110. 字符串接龙 学习: 代码随想录公开讲解 问题: 字典 strList 中从字符串 beginStr 和 endStr 的转换序列是一个按下述规格形成的序…...

华硕FL8000U加装16G+32G=48G内存条

华硕FL8000U加装16G32G48G内存条 一、华硕FL8000U加装内存条endl 一、华硕FL8000U加装内存条 相关视频链接: https://www.bilibili.com/video/BV1gw4dePED8/ endl...

前后端联调实战指南:Axios拦截器、CORS与JWT身份验证全解析

前言 在现代Web开发中,前后端分离架构已成为主流,而前后端联调则是开发过程中不可避免的关键环节。本文将深入探讨前后端联调中的三大核心技术:Axios拦截器的灵活运用、CORS跨域问题的全面解决方案以及JWT身份验证的安全实现。通过本文&…...

java高级 -Junit单元测试

Junit单元测试就是针对最小的功能:方法,编写测试代码对其进行正确性测试。用main方法进行测试的弊端是一个方法测试失败可能会影响别的方法的测试,也无法得到测试报告,需要我们自己观察数据是否正确。 此时,我们就需要…...

在 UVM验证环境中,验证 Out-of-Order或 Interleaving机制

在 UVM验证环境中,验证 Out-of-Order或 Interleaving机制 摘要:在 UVM (Universal Verification Methodology) 验证环境中,验证 Out-of-Order (乱序) 或 Interleaving (交错) 机制是验证复杂 SoC (System on Chip) 设计的重要任务,尤其是在验证高速接口(如 PCIe、AXI)、缓…...

V9数据库替换授权

文章目录 环境文档用途详细信息 环境 系统平台:Linux x86-64 Red Hat Enterprise Linux 7 版本:9.0 文档用途 1、本文档用于指导V9数据库替换授权。 2、V9数据库授权文件为license.dat。 详细信息 1、上传新的授权文件到服务器并修改授权文件属主为…...

勇闯Chromium—— Chromium的多进程架构

问题 构建一个永不崩溃或挂起的渲染引擎几乎是不可能的,构建一个绝对安全的渲染引擎也几乎是不可能的。 从某种程度上来说,2006 年左右的网络浏览器状态与过去单用户、协作式多任务操作系统的状况类似。正如在这样的操作系统中,一个行为不端的应用程序可能导致整个系统崩溃…...

Go语言中常量的命名规则详解

1. 常量的基本命名规则 1.1. 命名格式 1. 使用const关键字声明; 2. 命名格式:const 常量名 [类型] 值; 3. 类型可以省略,由编译器推断; 1.2. 命名风格 大小写规则: 1. 首字母大写:导出常…...

软件质量保证与测试实验

课程  软件质量保证与测试 目的&#xff1a;练习软件测试中白盒测试方法 内容&#xff1a; 测试如下程序段&#xff1a; #include <stdio.h>int main() {int i 1, n1 0, n2 0;float sum 0.0;float average;float score[100];printf("请输入分…...

历年华东师范大学保研上机真题

2025华东师范大学保研上机真题 2024华东师范大学保研上机真题 2023华东师范大学保研上机真题 在线测评链接&#xff1a;https://pgcode.cn/school?classification1 简单一位数代数式计算 题目描述 给一个小学生都会算的1位数与1位数运算的代数式&#xff0c;请你求出这个表…...

【C++】什么是静态库?什么是动态库?

静态库与动态库详解 静态库和动态库是软件开发中两种不同的代码共享和重用机制&#xff0c;它们在链接方式、内存使用和部署方式上有显著区别。 一、静态库(Static Library) 基本概念 静态库是在编译期间被完整复制到最终可执行文件中的预编译代码集合。 主要特点 链接时…...

项目阅读:Instruction Defense

总目录 大模型安全相关研究&#xff1a;https://blog.csdn.net/WhiffeYF/article/details/142132328 https://learnprompting.org/docs/prompt_hacking/defensive_measures/instruction https://www.doubao.com/chat/6945469301219586 速览 指令防御&#xff08;Instructio…...

springboot中拦截器配置使用

文章目录 前置拦截器代码拦截器注册疑问 前置 你使用 javaspringboot 常用在&#xff1a; 身份验证与授权&#xff0c;使用拦截器检查用户的身份验证状态和权限级别&#xff0c;确保只有经过验证且有适当权限的用户能够访问特定资源日志记录与审计性能分析与监控&#xff0…...

用 Python 构建自动驾驶的实时通信系统:让车辆“交流”起来!

用 Python 构建自动驾驶的实时通信系统:让车辆“交流”起来! 自动驾驶技术正加速变革全球交通体系,它不仅是机器学习与计算机视觉的胜利,更是一场 高效通信架构的革命。自动驾驶汽车需要实时交换信息,比如: 传感器数据(雷达、激光雷达、摄像头)V2V(车与车通信)V2X(…...

在机器学习中,L2正则化为什么能够缓过拟合?为何正则化等机制能够使一个“过度拟合训练集”的模型展现出更优的泛化性能?正则化

在现代机器学习的发展历程中&#xff0c;过拟合&#xff08;Overfitting&#xff09;始终是亟需克服的重要挑战。其表现如同在训练数据上构建过度复杂的映射函数&#xff0c;虽能实现近乎完美的拟合&#xff0c;但其泛化能力却显著受限&#xff0c;导致模型在测试集或实际应用中…...

day36 python神经网络训练

目录 一、数据准备与预处理 二、数据集划分与归一化 三、构建神经网络模型 四、定义损失函数和优化器 五、训练模型 六、评估模型 在机器学习和深度学习的实践中&#xff0c;信贷风险评估是一个非常重要的应用场景。通过构建神经网络模型&#xff0c;我们可以对客户的信用…...

k8s部署ELK补充篇:kubernetes-event-exporter收集Kubernetes集群中的事件

k8s部署ELK补充篇&#xff1a;kubernetes-event-exporter收集Kubernetes集群中的事件 文章目录 k8s部署ELK补充篇&#xff1a;kubernetes-event-exporter收集Kubernetes集群中的事件一、kubernetes-event-exporter简介二、kubernetes-event-exporter实战部署1. 创建Namespace&a…...

【Excel VBA 】窗体控件分类

一、Excel 窗体控件分类 Excel 中的窗体控件分为两大类型&#xff0c;适用于不同的开发需求&#xff1a; 类型所在选项卡特点表单控件开发工具 → 插入 → 表单控件简单易用&#xff0c;直接绑定宏&#xff0c;兼容性好&#xff0c;适合基础自动化操作。ActiveX 控件开发工具…...

C++性能相关的部分内容

C性能相关的部分内容 与底层硬件紧密结合 大端存储和小端存储&#xff08;硬件概念&#xff09; C在不同硬件上运行的结果可能不同 比如&#xff1a;输入01234567&#xff0c;对于大端存储的硬件会先在较大地址上先进行存储&#xff0c;而对于小端存储的硬件会先在较小地址上…...

Spring Boot 项目中常用的 ORM 框架 (JPA/Hibernate) 在性能方面有哪些需要注意的点?

在 Spring Boot 项目中使用 JPA (Java Persistence API) / Hibernate (作为 JPA 的默认实现) 时&#xff0c;性能是一个非常关键的考量点。虽然 ORM 极大地简化了数据库交互&#xff0c;但如果不注意&#xff0c;很容易引入性能瓶颈。以下是一些关键的性能注意事项&#xff1a;…...

基于大模型的大肠癌全流程预测与诊疗方案研究报告

目录 一、引言 1.1 研究背景与意义 1.2 研究目的与创新点 二、大模型技术概述 2.1 大模型原理与架构 2.2 大模型在医疗领域的应用现状 三、术前风险预测与准备 3.1 术前风险预测指标 3.2 大模型预测方法与结果 3.3 基于预测结果的术前准备方案 四、术中风险预测与应…...

解决DeepSeek部署难题:提升效率与稳定性的关键策略

DeepSeek 部署中常见问题及对应解决方案 随着大模型技术的快速发展&#xff0c;DeepSeek 作为国内领先的大语言模型之一&#xff0c;广泛应用于自然语言处理、智能客服、内容生成等多个领域。 然而&#xff0c;在实际部署过程中&#xff0c;许多开发者和企业会遇到一系列挑战&a…...

AI进行提问、改写、生图、联网搜索资料,嘎嘎方便!

极客侧边栏-AI板块 目前插件内已接入DeepSeek-R1满血版、Qwen3满血版 、豆包/智谱最新发布的推理模型以及各种顶尖AI大模型&#xff0c;并且目前全都可以免费不限次数使用&#xff0c;秒回不卡顿&#xff0c;联网效果超好&#xff01; 相比于市面上很多AI产品&#xff0c;极客…...

GStreamer开发笔记(四):ubuntu搭建GStreamer基础开发环境以及基础Demo

若该文为原创文章&#xff0c;转载请注明原文出处 本文章博客地址&#xff1a;https://blog.csdn.net/qq21497936/article/details/147714800 长沙红胖子Qt&#xff08;长沙创微智科&#xff09;博文大全&#xff1a;开发技术集合&#xff08;包含Qt实用技术、树莓派、三维、O…...

2021年认证杯SPSSPRO杯数学建模A题(第二阶段)医学图像的配准全过程文档及程序

2021年认证杯SPSSPRO杯数学建模 A题 医学图像的配准 原题再现&#xff1a; 图像的配准是图像处理领域中的一个典型问题和技术难点&#xff0c;其目的在于比较或融合同一对象在不同条件下获取的图像。例如为了更好地综合多种信息来辨识不同组织或病变&#xff0c;医生可能使用…...

CV中常用Backbone-3:Clip/SAM原理以及代码操作

前面已经介绍了简单的视觉编码器&#xff0c;这里主要介绍多模态中使用比较多的两种backbone&#xff1a;1、Clip&#xff1b;2、SAM。对于这两个backbone简单介绍基本原理&#xff0c;主要是讨论使用这个backbone。 1、CV中常用Backbone-2&#xff1a;ConvNeXt模型详解 2、CV中…...

RPC 协议详解、案例分析与应用场景

一、RPC 协议原理详解 RPC 协议的核心目标是让开发者像调用本地函数一样调用远程服务&#xff0c;其实现过程涉及多个关键组件与流程。 &#xff08;一&#xff09;核心组件 客户端&#xff08;Client&#xff09;&#xff1a;发起远程过程调用的一方&#xff0c;它并不关心调…...