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

JAVA常用分布式锁Redisson

1. 加锁过程

底层命令与数据结构
  • Redis 数据结构:使用 Hash 结构存储锁信息,Key 为锁名称,Field 为客户端唯一标识(如 UUID + 线程ID),Value 为锁的重入次数。

  • Lua 脚本原子性:通过 Lua 脚本在 Redis 中原子性执行加锁逻辑:

    
    if (redis.call('exists', KEYS[1]) == 0) thenredis.call('hincrby', KEYS[1], ARGV[2], 1);redis.call('pexpire', KEYS[1], ARGV[1]);return nil;
    end;
    if (redis.call('hexists', KEYS[1], ARGV[2]) == 1) thenredis.call('hincrby', KEYS[1], ARGV[2], 1);redis.call('pexpire', KEYS[1], ARGV[1]);return nil;
    end;
    return redis.call('pttl', KEYS[1]);
    • 若锁不存在(exists 为 0)或属于当前线程(hexists 为 1),则增加重入次数并刷新过期时间。

    • 若锁被其他线程占用,返回锁的剩余生存时间(TTL)。

可重入性
  • 同一线程多次获取锁时,重入次数递增,确保不会因多次加锁导致死锁。


2. 锁自动续期(Watchdog 机制)

  • 后台线程续期:加锁成功后,启动一个 Watchdog 线程(看门狗),定期(默认每 10 秒)检查锁是否仍被持有。

  • 续期条件:仅当客户端仍持有锁且业务未完成时,通过 pexpire 命令将锁的过期时间重置为初始值(默认 30 秒)。

  • 崩溃容错:若客户端崩溃,Watchdog 线程停止,锁最终因过期自动释放,避免死锁。


3. 释放锁

释放逻辑
  • Lua 脚本原子释放

    if (redis.call('hexists', KEYS[1], ARGV[3]) == 0) thenreturn nil;
    end;
    local counter = redis.call('hincrby', KEYS[1], ARGV[3], -1);
    if (counter > 0) thenredis.call('pexpire', KEYS[1], ARGV[2]);return 0;
    elseredis.call('del', KEYS[1]);redis.call('publish', KEYS[2], ARGV[1]);return 1;
    end;
    return nil;
    • 减少重入次数,若次数归零则删除锁,并通过 Pub/Sub 通知等待线程。

    • 确保只有锁的持有者能释放锁,避免误删。


4. 锁竞争与等待

  • 自旋重试:若锁被占用,客户端进入循环,间隔性尝试加锁。

  • Pub/Sub 订阅通知:通过订阅锁释放事件(redisChannel),避免频繁轮询。当锁释放时,Redis 发布消息通知等待线程竞争锁,减少无效请求。


5. 高可用与容错

Redis 部署模式
  • 单节点模式:简单但存在单点故障风险。

  • 主从/集群模式:使用 RedissonMultiLock 实现 RedLock 算法(需多个独立 Redis 节点):

    1. 向所有节点顺序申请锁。

    2. 当多数节点加锁成功且总耗时小于锁超时时间时,认为加锁成功。

    3. 规避主从切换导致锁丢失的问题,但需权衡性能和一致性。


6. 关键注意事项

  • 业务执行时间:业务逻辑必须在锁的过期时间内完成,否则锁可能提前释放。

  • 时钟同步问题:在 RedLock 中,若 Redis 节点间时钟不同步,可能导致锁失效。

  • 网络延迟:极端情况下,锁可能被多个客户端同时持有(需结合业务幂等性处理)。


总结

Redisson 分布式锁通过 Lua 脚本的原子性可重入设计Watchdog 自动续期和 Pub/Sub 通知机制,实现了高效的分布式锁管理。其核心优势在于:

  • 避免误删锁(仅持有者可释放)。

  • 支持可重入,适应复杂业务逻辑。

  • 自动续期防止业务未完成时锁过期。

  • 通过 RedLock 支持高可用场景,但需谨慎权衡一致性与性能。

http://www.xdnf.cn/news/1901.html

相关文章:

  • 大模型驱动智能服务变革:从全流程赋能到行业纵深落地
  • WHAT - 前端开发书单推荐
  • 带宽?增益带宽积?压摆率?
  • 基于物联网的智能家居安全防护系统设计
  • Java 24 深度解析:云原生时代的性能更新与安全重构
  • 用 Python 打造打篮球字符动画!控制台彩色炫酷输出,抖音搞怪视频灵感还原
  • 基于 Python(selenium) 的今日头条定向爬虫:根据输入的关键词在今日头条上进行搜索,并爬取新闻详情页的内容
  • 大型超市仓储管理5大痛点解析 智能穿梭车如何实现降本增效?
  • 数字后端设计 (五):布线——芯片里的「交通总动员」
  • [自记录]一次Nvidia显卡的AI容器基础镜像制作过程(含Torch版本和ONNXRuntime版本选择)
  • AI新战局:Gemini 2.5 Pro强势挑战OpenAI o3,谁是真“全能”?“锯齿AGI”时代已来临?
  • 快速了解redis,个人笔记
  • CRM管理优化的7个关键指标:提升客户留存率的科学方法
  • 优化算法
  • 文档编辑:reStructuredText全面使用指南 — 第二部分 基础语法
  • 【金仓数据库征文】-《深入探索金仓数据库:从基础到实战》
  • 【贝叶斯定理01】白话贝叶斯(原理篇)
  • MYSQL 常用数值函数 和 条件函数 详解
  • 1、RabbitMQ的概述笔记
  • Linux-06 ubuntu 系统截图软件使用简单记录
  • 百度Create2025 AI开发者大会:模型与应用的未来已来
  • 数智飞轮:AI时代企业增长的核心密码
  • 《免费开放”双刃剑:字节跳动Coze如何撬动AI生态霸权与暗涌危机?》
  • 最火向量数据库Milvus安装使用一条龙!
  • 2025 Java 开发避坑指南:如何避免踩依赖管理的坑?
  • 【C语言练习】003. 声明不同数据类型的变量并赋值
  • 不同ECU(MCU/ZCU/CCU)其部署(实现)的功能存在差异
  • 吞吐量,响应速率,占用内存大小是什么,用你自己的理解说一下
  • 基于51单片机的超声波液位测量与控制系统
  • 【AI】[特殊字符]生产规模的向量数据库 Pinecone 使用指南