分布式锁实战:Redis与Redisson的深度解析

一、分布式锁的必要性

在分布式系统中,当多个节点需要对共享资源进行读写操作时,传统的本地锁(如Java的synchronized或ReentrantLock)无法跨节点生效。此时,必须引入分布式锁来保证操作的原子性和一致性。分布式锁需满足以下核心特性:

  1. 互斥性:任意时刻仅一个客户端持有锁
  2. 防死锁:即使持有锁的客户端崩溃,锁仍可被释放
  3. 可重入性:同一客户端可多次获取同一把锁
  4. 一致性:解锁操作必须由锁的持有者执行

二、Redis分布式锁实现

Redis实现分布式锁主要利用Redis的setnx命令。
SETNX key value 是 Redis 的原子性命令,用于设置键值对,仅当键不存在时生效,否则不执行任何操作。

1. 基础实现(SETNX+EXPIRE)

// 加锁(非原子操作)
if (redisTemplate.execute((RedisCallback<Long>) connection -> connection.setNX(lockKey.getBytes(), UUID.randomUUID().toString().getBytes())) == 1) {redisTemplate.expire(lockKey, expireTime, TimeUnit.SECONDS);return true;
}
return false;
  • 问题:SETNX与EXPIRE的非原子性导致可能存在锁未设置过期时间的风险

2. 原子化实现(SET命令增强)

// 原子化加锁
Boolean success = redisTemplate.execute((RedisCallback<Boolean>) connection -> connection.set(lockKey.getBytes(), UUID.randomUUID().toString().getBytes(), Expiration.seconds(expireTime), RedisStringCommands.SetOption.SET_IF_ABSENT));
return success != null && success;
  • 优势:通过SET命令的NX选项实现原子性加锁与过期时间设置

3. 解锁实现

// 解锁(需验证锁归属)
if (UUID.fromString(redisTemplate.opsForValue().get(lockKey)).equals(currentUuid)) {redisTemplate.delete(lockKey);
}

三、Redis锁的缺陷与优化

1. 锁过期问题

  • 现象:业务执行时间超过锁过期时间,导致锁提前释放
  • 解决方案
    • 动态调整过期时间
    • 使用Redisson的看门狗机制

2. 锁误删风险

  • 现象:客户端A的锁过期后,客户端B获取锁并被客户端A误删
  • 解决方案
    // 使用Lua脚本保证原子性
    String script = "if redis.call('get', KEYS[1]) == ARGV[1] then return redis.call('del', KEYS[1]) else return 0 end";
    redisTemplate.execute(new DefaultRedisScript<>(script, Long.class), Arrays.asList(lockKey), currentUuid);
    

四、Redisson分布式锁进阶

1. 加锁过程

当一个线程尝试获取分布式锁时,Redisson 会执行以下操作:

  • 原子操作:使用 Redis 的 SETNX(SET if Not eXists)命令来尝试设置一个键值对,键表示锁的名称,值表示持有锁的线程标识(通常是一个 UUID)。如果设置成功,说明该线程成功获取到锁;如果设置失败,说明锁已经被其他线程持有。
  • 设置过期时间:为了避免锁被永久持有(例如持有锁的线程崩溃),Redisson 会为锁设置一个过期时间。可以使用 SET 命令的 NXEX 选项来原子性地完成设置键值对和过期时间的操作。
  • 锁重入:Redisson 支持锁重入,即同一个线程可以多次获取同一把锁而不会死锁。为了实现锁重入,Redisson 在 Redis 中存储的键值对的值是一个计数器,每次线程获取锁时计数器加 1,释放锁时计数器减 1,当计数器为 0 时才真正释放锁。

2. 锁续期机制

为了防止在业务逻辑执行期间锁过期,Redisson 引入了锁续期机制,也称为“看门狗”机制:

  • 定时任务:当线程成功获取锁后,Redisson 会启动一个定时任务,该任务会在锁过期时间的三分之一处执行,尝试对锁进行续期。
  • Lua 脚本:续期操作使用 Lua 脚本来保证原子性,脚本会检查锁的持有者是否还是当前线程,如果是则更新锁的过期时间。

3. 释放锁过程

当线程完成业务逻辑后,需要释放锁,Redisson 会执行以下操作:

  • 计数器减 1:如果是锁重入的情况,先将计数器减 1。
  • 释放锁:当计数器为 0 时,使用 Lua 脚本来删除 Redis 中的键值对,从而释放锁。Lua 脚本可以保证删除操作的原子性,避免在删除过程中出现并发问题。

示例代码

以下是一个简单的 Java 代码示例,展示了如何使用 Redisson 获取和释放分布式锁:

import org.redisson.Redisson;
import org.redisson.api.RLock;
import org.redisson.api.RedissonClient;
import org.redisson.config.Config;import java.util.concurrent.TimeUnit;public class RedissonLockExample {public static void main(String[] args) {// 创建 Redisson 客户端Config config = new Config();config.useSingleServer().setAddress("redis://127.0.0.1:6379");RedissonClient redisson = Redisson.create(config);// 获取锁RLock lock = redisson.getLock("myLock");try {// 尝试获取锁,等待 10 秒,锁的过期时间为 30 秒boolean isLocked = lock.tryLock(10, 30, TimeUnit.SECONDS);if (isLocked) {// 模拟业务逻辑System.out.println("获取到锁,开始执行业务逻辑");Thread.sleep(5000);System.out.println("业务逻辑执行完毕");}} catch (InterruptedException e) {e.printStackTrace();} finally {// 释放锁if (lock.isHeldByCurrentThread()) {lock.unlock();System.out.println("锁已释放");}}// 关闭 Redisson 客户端redisson.shutdown();}
}

4. 锁模式对比

锁类型特性适用场景
公平锁按等待顺序分配锁高并发有序场景
联锁多个独立Redis节点的组合锁金融级安全场景
红锁多数节点达成共识的锁机制分布式系统强一致性要求

五、最佳实践建议

  1. 锁粒度控制:避免粗粒度锁,优先使用细粒度锁
  2. 过期时间设置:根据业务耗时合理设置,建议30-60秒
  3. 异常处理:所有加锁操作必须包含finally块释放锁
  4. 监控报警:对锁竞争、锁超时等异常情况进行监控
  5. 降级策略:锁获取失败时要有回退机制,避免系统雪崩

六、总结

Redis原生锁与Redisson框架为分布式锁提供了不同层级的解决方案:

  • Redis原生方案适用于轻量级场景,需关注原子性与锁过期问题
  • Redisson框架通过自动续期、多种锁模式等特性,提供了企业级的分布式锁解决方案

在实际应用中,应根据系统规模、一致性要求和业务特性选择合适的实现方式,同时结合监控和报警机制保障系统的稳定性。

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

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

相关文章

Dify本地安装部署笔记

目录 方式一【docker安装】&#xff1a; 步骤 1&#xff1a;准备工作 步骤2: 克隆dify仓库 步骤3:部署启动dify 步骤 4&#xff1a;访问 Dify 步骤5:升级dify 方式二【源码安装】&#xff1a; 步骤1. 硬件&#xff1a;最低安装要求 步骤2: 业务服务前的3个服务 1. 安…

质检LIMS系统在食品生产加工企业的应用 如何保证食品生产企业的安全

在食品生产加工领域&#xff0c;质量安全是贯穿全产业链的生命线。随着《食品安全法》对全过程追溯要求的深化&#xff0c;传统实验室管理模式已难以满足高效、精准的质量管控需求。质检实验室信息管理系统&#xff08;LIMS&#xff09;作为数字化升级的核心工具&#xff0c;正…

自动驾驶VLA模型技术解析与模型设计

1.前言 2025年被称为“VLA上车元年”&#xff0c;以视觉语言动作模型&#xff08;Vision-Language-Action Model, VLA&#xff09;为核心的技术范式正在重塑智能驾驶行业。VLA不仅融合了视觉语言模型&#xff08;VLM&#xff09;的感知能力和端到端模型的决策能力&#xff0c;…

UDP套接字编程(代码)

什么是socket套接字编程&#xff1f; 通过Ip地址 端口号这种方式定位一台主机&#xff0c;这样的方式我们就叫做socket套接字。 Udp Socket 接口介绍 这些案列我们使用的接口基本都是一样的&#xff0c;所以在这里我先把接口介绍完&#xff0c;具体的细节后面在说明。 创…

汽车行业可信数据空间研究探索

近期&#xff0c;相关老师在新能源汽车国家大数据联盟微课堂发表了题为“汽车行业可信数据空间研究探索”的演讲&#xff0c;主要包括可信数据空间的概念内涵、汽车行业可信数据空间的发展现状、数据流通场景和技术需求研究、汽车行业可信数据空间的场景建设建议四个方面展开。…

圆弧插补相关算法汇总(C++和ST源代码)

运动控制需要了解相关的插补概念,在阅读本篇博客之前需要了解相关的准备知识,常用链接如下: SMART PLC直线插补详解-CSDN博客文章浏览阅读2.1k次,点赞2次,收藏4次。本文介绍了SMART PLC中轴组对象的概念,详细讲解了直线插补的原理和指令使用,包括SMART PLC从V2.7版本开…

Entity Framework框架

深入理解C#中的Entity Framework框架&#xff1a;从理论到实践 在C#开发中&#xff0c;与数据库交互是几乎所有应用程序的核心需求之一。Entity Framework (EF) 作为微软官方推出的ORM框架&#xff0c;极大地简化了数据库操作。本文将带您深入理解EF框架的核心概念&#xff0c…

C++11QT复习 (五)

文章目录 **Day6-2 成员访问运算符重载&#xff08;2025.03.25&#xff09;****1. 复习****2. 成员访问运算符重载****2.1 箭头运算符 (->) 重载****(1) 语法** **2.2 解引用运算符 (*) 重载****(1) 语法** **3. 代码分析****3.1 代码结构****3.2 代码解析****(1) Data 类**…

简历含金量的描述和注意事项!

背景 最近&#xff0c;在公司负责后端相关面试&#xff0c;简历看了不下 50 份&#xff0c;面试 10&#xff0c;纯手码 2000 多字&#xff0c;说说我对简历的看法&#xff0c;希望给大家一点启发。 教育经历 在众多求职面试中&#xff0c;我发现多数求职者容易忽视教育背景的…

cellnet框架概述

cellnet框架是一个‌高性能、组件化、多协议支持‌的开源服务器网络库&#xff0c;专注于游戏服务器、分布式的多进程通信等场景的开发。 一、核心特性 ‌支持多个主流协议&#xff0c;包括TCP、UDP、HTTP、WebSocket。并且抽象底层协议差异&#xff0c;统一网络连接管理‌。 …

【加密社】如何创建自己的币圈工具站

需要准备的工作 1.域名 2.服务器 周末的时候主要弄了快讯这方面的代码 我这里用的是星球日报的api&#xff0c;也可以订阅他们的rss&#xff0c;这部分在github上是开源的 https://github.com/ODAILY 我这里用的是WordPressonenav主题&#xff0c;然后用小工具在主页展示&am…

Docker学习笔记(十一)宿主机无法链接宿主机问题处理

故障排查优先级排序 服务状态 → 2. 端口监听 → 3. 防火墙 → 4. 权限配置 → 5. 网络路由 &#xff08;按此顺序可覆盖95%的常见问题‌15&#xff09; mysql镜像启动命令&#xff1a; docker run -p 3306:3306 --restartalways --name mysqlv8 -e MYSQL_ROOT_PASSWORDCd…

力扣:回溯算法

组合I class Solution {List<List<Integer>> result new ArrayList(); // 所有结果集List<Integer> list new ArrayList(); // 当前结果集public List<List<Integer>> combine(int n, int k) {dfs(n, k, 1);return result;}public void dfs(i…

华为HCIE鸿蒙应用开发认证靠谱吗?

在万物互联时代&#xff0c;智能终端设备的多样性与协同需求催生了操作系统的革新。华为HarmonyOS&#xff08;鸿蒙系统&#xff09;凭借其分布式架构与全场景能力&#xff0c;正成为打破设备边界、重塑用户体验的核心技术底座。HCIE鸿蒙应用开发认证作为华为认证体系的顶级资质…

23种设计模式-原型(Prototype)设计模式

原型设计模式 &#x1f6a9;什么是原型设计模式&#xff1f;&#x1f6a9;原型设计模式的特点&#x1f6a9;原型设计模式的结构&#x1f6a9;原型设计模式的优缺点&#x1f6a9;原型设计模式的Java实现&#x1f6a9;代码总结&#x1f6a9;总结 &#x1f6a9;什么是原型设计模式…

Oracle-rman restore遭遇RMAN-03002与ORA-19563

文章目录 在原DB上检查是否有重复的文件名&#xff1a;查看rman恢复的日志修正重名部分重新执行rman恢复结论&#xff1a; 在 RMAN 恢复过程中&#xff0c;遇到RMAN-03002连同ORA-19563:错误。 操作是将 Oracle 10.0.5的数据库备份从 RMAN備份恢复到另一台测试主机的同一个目录…

运维网络排查工具介绍与使用

作为一名运维工程师&#xff0c;日常工作中最令人头疼的莫过于各种网络故障。在过去一年半的运维生涯中&#xff0c;我积累了丰富的网络故障排查经验&#xff0c;今天就来和大家分享一下如何运用抓包工具&#xff08;Wireshark、tcpdump&#xff09;和网络排查工具&#xff08;…

解决vscode终端和本地终端python版本不一致的问题

&#x1f33f; 问题描述 本地终端&#xff1a; vscode终端&#xff1a; 别被这个给骗了&#xff0c;继续往下看&#xff1a; 难怪我导入一些包的时候老提示找不到&#xff0c;在本地终端就不会这样&#xff0c;于是我严重怀疑vscode中的python版本和终端不一样&#xff0c…

Sublime全局搜索快捷键Ctrl+Shift+F不能使用解决

问题描述&#xff1a; 在安装好Sublime后&#xff0c;我们使用快捷键进行全局搜索&#xff0c;发现没有反应&#xff0c;但是中文输入变成了繁体。 解决方案&#xff1a; 如截图&#xff0c;在关闭简繁切换的快捷键或者换成其他的就行

海康HTTP监听报警事件数据

http监听接收报警事件数据 海康获取设备报警事件数据两种方式&#xff1a; 1、sdk 布防监听报警事件数据,服务端布防。&#xff08;前面文章有示例&#xff09; 2、http监听接收报警事件数据&#xff0c;设备直接推送。 http监听接收报警事件数据&#xff0c;服务端可以使用n…