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

Redisson分布式锁深度解析:原理与实现机制

Redisson作为Redis Java客户端中的分布式解决方案佼佼者,其分布式锁实现被广泛应用于生产环境。以下从底层设计到源码实现进行全面剖析。

一、核心架构设计

1. 整体架构图

graph LRA[客户端] --> B[RLock接口]B --> C[RedissonLock]C --> D[Redis命令执行]D --> E[Lua脚本]E --> F[Redis集群]C --> G[WatchDog]G --> C

2. 核心组件

组件

作用

关键特性

RLock接口

定义锁行为规范

扩展Java Lock接口

RedissonLock

基础锁实现

封装Redis交互逻辑

Lua脚本

原子操作执行

保证操作原子性

WatchDog

锁续期守护线程

防止业务未完成锁过期

PubSub

锁释放通知

实现阻塞等待机制

二、加锁流程深度解析

1. 加锁核心Lua脚本

-- KEYS[1]: 锁key名
-- ARGS[1]: 锁超时时间(毫秒)
-- ARGS[2]: 客户端唯一标识(UUID+线程ID)-- 1. 检查锁是否存在
if (redis.call('exists', KEYS[1]) == 0) then-- 2. 不存在则加锁redis.call('hset', KEYS[1], ARGV[2], 1);-- 3. 设置过期时间redis.call('pexpire', KEYS[1], ARGV[1]);return nil;
end;-- 4. 锁已存在,检查是否当前线程持有
if (redis.call('hexists', KEYS[1], ARGV[2]) == 1) then-- 5. 重入次数+1redis.call('hincrby', KEYS[1], ARGV[2], 1);-- 6. 刷新过期时间redis.call('pexpire', KEYS[1], ARGV[1]);return nil;
end;-- 7. 返回剩余生存时间(毫秒)
return redis.call('pttl', KEYS[1]);

2. 加锁流程时序图

sequenceDiagramparticipant Clientparticipant Redissonparticipant RedisClient->>Redisson: tryLock(10, TimeUnit.SECONDS)Redisson->>Redis: 执行Lua脚本alt 锁不存在或可重入Redis-->>Redisson: 加锁成功Redisson->>Redisson: 启动WatchDogRedisson-->>Client: 返回trueelse 锁被占用Redis-->>Redisson: 返回TTLRedisson->>Redis: 订阅锁释放频道Redis--xRedisson: 等待通知(异步)Redisson->>Client: 返回false或阻塞end

3. 关键参数说明

参数

作用

推荐设置

lockWatchdogTimeout

WatchDog检查间隔

30000ms(默认)

leaseTime

锁持有时间

不设置(启用WatchDog)

waitTime

获取锁等待时间

根据业务调整

三、锁续期机制(WatchDog)

1. 实现原理

// RedissonLock类中
protected void scheduleExpirationRenewal(String threadId) {// 每10秒执行一次续期Timeout task = commandExecutor.getConnectionManager().newTimeout(new TimerTask() {public void run(Timeout timeout) {// 执行续期Lua脚本RFuture<Boolean> future = renewExpirationAsync(threadId);future.onComplete((res, e) -> {if (e != null) {return;}if (res) {// 递归调用实现定期执行scheduleExpirationRenewal(threadId);}});}}, internalLockLeaseTime / 3, TimeUnit.MILLISECONDS);
}

2. 续期Lua脚本

-- KEYS[1]: 锁key
-- ARGV[1]: 续期时间
-- ARGV[2]: 客户端标识if (redis.call('hexists', KEYS[1], ARGV[2]) == 1) thenredis.call('pexpire', KEYS[1], ARGV[1]);return 1;
end;
return 0;

3. 异常处理机制

  • 网络中断:重试3次后放弃续期
  • Redis故障:通过ConnectionListener检测连接状态
  • 客户端崩溃:依赖Redis的key自动过期

四、解锁机制剖析

1. 解锁Lua脚本

lua

-- KEYS[1]: 锁key
-- KEYS[2]: 锁释放频道
-- ARGV[1]: 释放消息
-- ARGV[2]: 客户端标识
-- ARGV[3]: 锁超时时间-- 1. 检查锁是否存在
if (redis.call('hexists', KEYS[1], ARGV[2]) == 0) thenreturn nil;
end;-- 2. 减少重入计数
local counter = redis.call('hincrby', KEYS[1], ARGV[2], -1);
if (counter > 0) then-- 3. 仍有重入则刷新过期时间redis.call('pexpire', KEYS[1], ARGV[3]);return 0;
else-- 4. 完全释放锁redis.call('del', KEYS[1]);-- 5. 发布释放消息redis.call('publish', KEYS[2], ARGV[1]);return 1;
end;
return nil;

2. 解锁流程

  1. 检查当前线程是否持有锁
  2. 减少重入计数器
  3. 计数器归零时删除key
  4. 通过PubSub发布解锁消息
  5. 取消WatchDog续期任务

五、高级特性实现

1. 公平锁实现

RLock fairLock = redisson.getFairLock("fairLock");

实现原理

  1. 使用Redis列表维护等待线程队列
  2. 通过ZSet记录线程超时时间
  3. 采用"先到先服务"策略

2. 读写锁

RReadWriteLock rwLock = redisson.getReadWriteLock("rwLock");
rwLock.readLock().lock();
rwLock.writeLock().lock();

存储结构

# 写锁
rwLock: { "mode": "write", "UUID_01:threadId_1": 1 }# 读锁
rwLock: { "mode": "read", "UUID_01:threadId_1": 3, "UUID_02:threadId_1": 1 }

3. 联锁(MultiLock)

RLock lock1 = redisson.getLock("lock1");
RLock lock2 = redisson.getLock("lock2");
RLock multiLock = redisson.getMultiLock(lock1, lock2);

实现特点

  • 所有锁必须全部获取成功
  • 采用相同的leaseTime
  • 整体视为一个原子操作

六、生产环境最佳实践

1. 参数调优建议

Config config = new Config();
config.setLockWatchdogTimeout(30000); // 看门狗超时
config.useClusterServers().setPingConnectionInterval(5000) // 心跳检测.setTimeout(3000); // 命令超时

2. 异常处理模板

RLock lock = redisson.getLock("resourceLock");
try {if (lock.tryLock(5, 30, TimeUnit.SECONDS)) {try {// 业务逻辑} finally {lock.unlock();}}
} catch (InterruptedException e) {Thread.currentThread().interrupt();throw new BusinessException("锁获取中断");
} catch (Exception e) {// 记录监控指标monitor.recordLockFailure();throw e;
}

3. 监控指标采集

指标

采集方式

告警阈值

锁等待时间

Micrometer Timer

>1s

锁竞争频率

Redis慢查询日志

>50次/秒

锁续期失败

WatchDog日志

连续3次失败

锁泄漏

未释放锁统计

存活>10min

七、常见问题解决方案

1. 锁续期失败

场景:Redis节点故障导致续期中断
方案

  • 启用Redisson的retryInterval参数
  • 配置集群模式自动切换节点

2. 网络分区问题

场景:客户端与Redis集群断开连接
方案

  • 设置合理的timeout参数
  • 实现降级策略(如本地缓存)

3. 锁竞争激烈

场景:大量线程等待同一把锁
方案

  • 引入锁分段(如将锁拆分为lock_1, lock_2...)
  • 使用tryLock替代lock避免长时间阻塞

Redisson通过精心设计的Lua脚本、WatchDog机制和灵活的配置选项,提供了生产级的分布式锁实现。理解这些底层原理有助于在复杂分布式系统中正确使用和问题排查。

Redis 分布式锁实现深度解析-CSDN博客

相关文章:

Redisson分布式锁深度解析:原理与实现机制

Redisson作为Redis Java客户端中的分布式解决方案佼佼者&#xff0c;其分布式锁实现被广泛应用于生产环境。以下从底层设计到源码实现进行全面剖析。 一、核心架构设计 1. 整体架构图 graph LRA[客户端] --> B[RLock接口]B --> C[RedissonLock]C --> D[Redis命令执…...

Linux 系统调用实现机制详解

Linux 系统调用实现机制详解 —— fork()、execve()、waitpid() 内核层面的秘密 在 Linux 内核中&#xff0c;fork()、execve() 和 waitpid() 是构建多任务操作系统的三大基石&#xff0c;它们涉及进程控制、内存管理、文件系统等多个子系统。本文将带你一探它们在 内核层面的…...

责任链模式_行为型_GOF23

责任链模式 责任链模式&#xff08;Chain of Responsibility Pattern&#xff09;是一种行为型设计模式&#xff0c;核心思想是将多个处理请求的对象连成一条链&#xff0c;请求沿链传递直到被处理。它像现实中的“多级审批流程”——请假或报销时&#xff0c;申请会逐级提交给…...

03-SpringBoot3入门-配置文件(自定义配置及读取)

1、自定义配置 # 自定义配置 zbj:user:username: rootpassword: 123456# 自定义集合gfs:- a- b- c2、读取 1&#xff09;User类 package com.sgu.pojo;import lombok.Data; import org.springframework.boot.context.properties.ConfigurationProperties; import org.spring…...

学习记录-软件测试基础

一、软件测试分类 1.按阶段&#xff1a;单元测试&#xff08;一般开发自测&#xff09;、集成测试、系统测试、验收测试 2.按代码可见度测试&#xff1a;黑盒测试、灰盒测试、白盒测试 3.其他&#xff1a;冒烟测试(冒烟测试主要是在开发提测后进行&#xff0c;主要是测试主流…...

【蓝桥杯每日一题】3.28

&#x1f3dd;️专栏&#xff1a; 【蓝桥杯备篇】 &#x1f305;主页&#xff1a; f狐o狸x "今天熬的夜&#xff0c;会变成明天奖状的闪光点&#xff01;" 目录 一、唯一的雪花 题目链接 题目描述 解题思路 解题代码 二、逛画展 题目链接 题目描述 解题思路 解题代…...

优秀的 React 入门开源项目推荐

以下是一些优秀的 React 入门开源项目推荐&#xff0c;涵盖不同应用场景和功能模块&#xff0c;适合学习和实践&#xff1a; 1. Jira Clone 仓库地址&#xff1a;GitHub - oldboyxx/jira_clone 亮点&#xff1a; 基于 React Hooks 实现&#xff0c;模仿 Jira 的任务管理功能。…...

万字长文详解Text-to-SQL

什么是Text-to-SQL 在各个企业数据量暴涨的现在&#xff0c;Text-to-SQL越来越重要了&#xff0c;所以今天就来聊聊Text-to-SQL。 Text-to-SQL是一种将自然语言查询转换为数据库查询的技术。它可以让用户通过自然语言来查询数据库&#xff0c;而不需要编写复杂的SQL语句。 T…...

【Linux】动静态库的制作与使用

一.对软硬链接的补充 1、无法对目录进行硬链接 为什么呢&#xff1f; 首先&#xff0c;我们在访问文件时&#xff0c;每一个文件都会有自己的dentry结构&#xff0c;这些结构会在内存中维护一棵路径树&#xff0c;来快速进行路径查找。但是如果某个节点直接使用硬链接到了根节…...

ubuntu22.04 如何安装 ch341 驱动

前言 本篇是介绍ubuntu22.04如何安装 ch341 驱动&#xff0c;并对其中遇到的问题进行整理。 一、流程 1.1 查看CH340驱动 首先是查看ubuntu22.04系统自带的驱动&#xff0c;用以下命令即可 ls /lib/modules/$(uname -r)/kernel/drivers/usb/serial 然后会跳出以下界面&…...

个人博客网站从搭建到上线教程

步骤1:设计个人网站 设计个人博客网站的风格样式,可以在各个模板网站上多浏览浏览,以便有更多设计网站风格样式的经验。 设计个人博客网站的内容,你希望你的网站包含哪些内容如你的个人基本信息介绍、你想分享的项目、你想分享的技术文档等等。 步骤2:选择开发技术栈 因…...

android 一步完成 aab 安装到手机

家人们谁懂&#xff01;在 Android 系统安装 aab 应用超麻烦。满心期待快速体验&#xff0c;却发现 aab 无法直装&#xff0c;得先转为 apks 格式&#xff0c;这过程复杂易错。好不容易转好&#xff0c;还得安装 apks&#xff0c;一番折腾&#xff0c;时间与耐心全耗尽。别愁&a…...

c#使用forms实现屏幕截图

说明&#xff1a; c#使用forms实现屏幕截图 step1: 点击按钮&#xff0c;拖拽&#xff0c;截图&#xff0c;保存本地 C:\Users\wangrusheng\RiderProjects\WinFormsApp1\WinFormsApp1\Form1.cs using System; using System.Drawing; using System.Drawing.Imaging; using Syst…...

mac m4 Homebrew安装MySQL 8.0

1.使用Homebrew安装MySQL8 在终端中输入以下命令来安装MySQL8&#xff1a; brew install mysql8.0 安装完成后&#xff0c;您可以通过以下命令来验证MySQL是否已成功安装&#xff1a; 2.配置mysql环境变量 find / -name mysql 2>/dev/null #找到mysql的安装位置 cd /op…...

UE5学习笔记 FPS游戏制作26 UE中的UI

文章目录 几个概念创建一个UI蓝图添加UI获取UI的引用 切换设计器和UI蓝图将UI添加到游戏场景锚点轴点slotSizeToContent三种UI数据更新方式(Text、Image)函数绑定属性绑定事件绑定 九宫格分割图片按钮设置图片绑定按下事件 下拉框创建添加数据修改样式常用函数 滚动框创建添加数…...

Navicat导出mysql数据库表结构说明到excel、word,单表导出方式记录

目前只找到一张一张表导出的方式 使用information_schema传入表名查询 字段名根据需要自行删减&#xff0c;一般保留序号、字段名、类型、说明就行 SELECT COLUMNS.ORDINAL_POSITION AS 序号, COLUMNS.COLUMN_NAME AS 字段名, COLUMNS.COLUMN_TYPE AS 类型(长度), COLUMNS.N…...

目标检测 AP 计算 实例 python

以下是使用 Python 实现目标检测中 ‌Average Precision (AP)‌ 计算的完整实例&#xff0c;包含代码和注释。这里以 ‌Pascal VOC 标准‌ 为例&#xff08;IoU阈值0.5&#xff09;。 步骤1&#xff1a;准备数据 假设&#xff1a; gt_boxes: 真实标注框列表&#xff0c;格式为 …...

HarmonyOS NEXT图形渲染体系:重新定义移动端视觉体验

一、革命性架构设计 1.1 多线程并行渲染引擎 HarmonyOS NEXT通过四级流水线并行架构实现渲染效率质的飞跃&#xff0c;其核心包含&#xff1a; 优先级任务调度器&#xff1a;动态分配紧急渲染任务&#xff08;如手势反馈&#xff09;与常规任务智能线程池管理&#xff1a;根…...

使用 Docker 18 安装 Eureka:解决新版本 Docker 不支持的问题

使用 Docker 18 安装 Eureka&#xff1a;解决新版本 Docker 不支持的问题 在微服务架构中&#xff0c;Eureka 是一个常用的注册中心&#xff0c;用于服务发现和管理。然而&#xff0c;随着 Docker 版本的更新&#xff0c;一些新版本的 Docker 对 Eureka 的支持并不友好。如果你…...

Linux驱动开发 中断处理

目录 序言 1.中断的概念 2.如何使用中断 中断处理流程 中断上下文限制 屏蔽中断/使能 关键区别与选择 上半部中断 下半部中断 软中断&#xff08;SoftIRQ&#xff09; 小任务(Tasklet) 工作队列&#xff08;Workqueue&#xff09; 线程 IRQ&#xff08;Threaded IRQ…...

Centos主机检查脚本

使用方法&#xff1a; 将脚本保存为 CentOS_syscheck.sh 添加执行权限&#xff1a; chmod x CentOS_syscheck.sh 执行脚本&#xff1a; ./CentOS_syscheck.sh #!/bin/bash# 设置颜色变量 RED\033[0;31m GREEN\033[0;32m YELLOW\033[0;33m BLUE\033[0;34m NC\033[0m # 重置…...

python系统之综合案例:用python打造智能诗词生成助手

不为失败找理由&#xff0c;只为成功找方法。所有的不甘&#xff0c;因为还心存梦想&#xff0c;所以在你放弃之前&#xff0c;好好拼一把&#xff0c;只怕心老&#xff0c;不怕路长。 python系列之综合案例 前言一、项目描述二、项目需求三、 项目实现1、开发准备2、代码实现 …...

【微服务】SpringBoot整合LangChain4j 操作AI大模型实战详解

目录 一、前言 二、Langchain4j概述 2.1 Langchain4j 介绍 2.1.1 Langchain4j 是什么 2.1.2 主要特点 2.2 Langchain4j 核心组件介绍 2.3 Langchain4j 核心优势 2.4 Langchain4j 核心应用场景 三、SpringBoot 整合 LangChain4j 组件使用 3.1 前置准备 3.1.1 获取apik…...

DeepSeek:巧用前沿AI技术,开启智能未来新篇章

引言 近年来&#xff0c;人工智能&#xff08;AI&#xff09;技术迅猛发展&#xff0c;大模型成为全球科技竞争的核心赛道。在这场AI革命中&#xff0c;DeepSeek作为中国领先的大模型研发团队&#xff0c;凭借其创新的技术架构、高效的训练方法和广泛的应用场景&#xff0c;迅…...

23种设计模式-结构型模式-桥接器

文章目录 简介问题解决方案示例总结 简介 桥接器是一种结构型设计模式&#xff0c;可将一个大类或一系列紧密相关的类拆分为抽象和实现两个独立的层次结构&#xff0c;从而能在开发时分别使用。 问题 假如你有一个几何形状Shape类&#xff0c;它有两个子类&#xff1a;圆形C…...

K8S学习之基础五十八:部署nexus服务

部署nexus服务 Nexus服务器是一个代码包管理的服务器&#xff0c;可以理解 Nexus 服务器是一个巨大的 Library 仓库。Nexus 可以支持管理的工具包括 Maven &#xff0c; npm 等&#xff0c;对于 JAVA 开发来说&#xff0c;只要用到 Maven 管理就可以了。Nexus服务器作用&#x…...

Docker Desktop 界面功能介绍

Docker Desktop 界面功能介绍 左侧导航栏 Containers(容器): 用于管理容器,包括查看运行中或已停止的容器,检查容器状态、日志,执行容器内命令,启动、停止、删除容器等操作。Images(镜像): 管理本地 Docker 镜像,可查看镜像列表、从 Docker Hub 拉取新镜像、删除镜…...

C++ set map

1.set和map是什么 set和map是 C STL 提供的容器&#xff0c;用于高效的查找数据&#xff0c;底层采用红黑树实现&#xff0c;其中set是Key模型&#xff0c;map是Key-Value模型 set和map的基本使用较为简单&#xff0c;这里不再叙述&#xff0c;直接进入实现环节 2.set和map的…...

Spring AI Alibaba 对话记忆使用

一、对话记忆 (ChatMemory)简介 1、对话记忆介绍 ”大模型的对话记忆”这一概念&#xff0c;根植于人工智能与自然语言处理领域&#xff0c;特别是针对具有深度学习能力的大型语言模型而言&#xff0c;它指的是模型在与用户进行交互式对话过程中&#xff0c;能够追踪、理解并利…...

Ubuntu24.04 离线安装 MySQL8.0.41

一、环境准备 1.1 官方下载MySQL8.0.41 完整包 1.2 上传包 & 解压 上传包名称是&#xff1a;mysql-server_8.0.41-1ubuntu24.04_amd64.deb-bundle.tar # 切换到上传目录 cd /home/MySQL8 # 解压&#xff1a; tar -xvf mysql-server_8.0.41-1ubuntu24.04_amd64.deb-bundl…...