搭建Redisson流程以及解读MutilLock源码解决分布式锁的主从一致性问题

news2025/1/12 3:05:52

搭建Redisson流程以及解读MutilLock源码解决分布式锁的主从一致性问题

    • 1、搭建3台独立主节点的redis服务
    • 2、创建java redisson客户端
    • 3、获取分布式锁
    • 4、分析获取锁源码
      • getMultiLock
      • tryLock(long waitTime, long leaseTime, TimeUnit unit)
    • 5、总结

1、搭建3台独立主节点的redis服务

为了方便,采用docker进行搭建

搭建前准备:使用docker容器搭建redis服务,不同于其他服务,启动后并没有找到有关redis的配置,但是其实docker 提供的redis镜像已经默认帮我们添加了许多配置,比如说通过命令直接搭建的redis服务,可以直接被其他服务器访问

docker run -d -p 6379:6379 --name redis_6379 redis

但是搭建过redis服务器的小伙伴都知道,必须在redis.conf配置文件中修改bind 0.0.0.0 才能被除本机外其他服务器所访问

现在我们是基于学习搭建的测试服务器,所以我们还是通过制定配置文件的方式启动

在这里插入图片描述

红色框部分不需要创建,只需要保证 /root/redis 多其他文件夹以及redis.conf文件即可

redis.conf配置文件

bind 0.0.0.0
appendonly yes

创建文件 vim docker_run_redis.sh

挂载redis.conf文件、以及aof、rbd的持久化文件,并且通过redis-server命令指定 目标文件启动

docker run -d -p 6379:6379 --name redis_zs -v /root/redis/redis.conf:/data/redis.conf  -v /root/redis/data:/data redis redis-server /data/redis.conf
docker run -d -p 6380:6379 --name redis_zs1 -v /root/redis/redis_zs1/redis.conf:/data/redis.conf  -v /root/redis/redis_zs1/data:/data redis redis-server /data/redis.conf
docker run -d -p 6381:6379 --name redis_zs2 -v /root/redis/redis_zs2/redis.conf:/data/redis.conf  -v /root/redis/redis_zs2/data:/data redis redis-server /data/redis.conf

sh docekr_cun_redis.sh 启动当前命令

查看redis服务节点启动情况
在这里插入图片描述

搭建完成

2、创建java redisson客户端

@Configuration
public class RedisConfig {
  @Bean
  public RedissonClient redissonClient() {

    // 配置
    Config config = new Config();
    config.useSingleServer().setAddress("redis://192.168.79.128:6379");
    // 创建 RedissonClient 对象
    return Redisson.create(config);
  }

  @Bean
  public RedissonClient redissonClient1() {

    // 配置
    Config config = new Config();
    config.useSingleServer().setAddress("redis://192.168.79.128:6380");
    // 创建 RedissonClient 对象
    return Redisson.create(config);
  }

  @Bean
  public RedissonClient redissonClient2() {

    // 配置
    Config config = new Config();
    config.useSingleServer().setAddress("redis://192.168.79.128:6381");
    // 创建 RedissonClient 对象
    return Redisson.create(config);
  }
}

3、获取分布式锁

@SpringBootTest
public class RedissonTest {

  @Resource
  private RedissonClient redissonClient;

  @Resource
  private RedissonClient redissonClient1;

  @Resource
  private RedissonClient redissonClient2;

  private RLock lock;

  @BeforeEach
  void setUp() {

    RLock lock1 = redissonClient.getLock("test");
    RLock lock2 = redissonClient1.getLock("test");
    RLock lock3 = redissonClient2.getLock("test");

    // 创建联锁
    lock = redissonClient.getMultiLock(lock1, lock2, lock3);
  }
  
  @Test
  void method1() throws InterruptedException {
    boolean isLock = lock.tryLock(1L, TimeUnit.SECONDS);
    if (!isLock) {
      log.error("获取锁失败 ... 1");
      return;
    }
    try {
      log.info("获取锁成功 ... 1");
      method2();
      log.info("开始执行业务 ... 1");
    } finally {
      log.warn("准备释放锁 ... 1");
      lock.unlock();
    }
  }

  @Test
  void method2() {
    boolean isLock = lock.tryLock();
    if (!isLock) {
      log.error("获取锁失败 ... 2");
      return;
    }
    try {
      log.info("获取锁成功 ... 2");
      log.info("开始执行业务 ... 2");
    } finally {
      log.warn("准备释放锁 ... 2");
      lock.unlock();
    }
  }
}

4、分析获取锁源码

getMultiLock

该方法入参是可变参数

最终赋值给RedissonMutilLock 的常量locks

@Override
public RLock getMultiLock(RLock... locks) {
    return new RedissonMultiLock(locks);
}
public RedissonMultiLock(RLock... locks) {
  if (locks.length == 0) {
    throw new IllegalArgumentException("Lock objects are not defined");
  }
  this.locks.addAll(Arrays.asList(locks));
}

tryLock(long waitTime, long leaseTime, TimeUnit unit)

@Override
public boolean tryLock(long waitTime, TimeUnit unit) throws InterruptedException {
  // 没有设置锁的过期释放时间默认leaseTime为-1
  return tryLock(waitTime, -1, unit);  
}

@Override
public boolean tryLock(long waitTime, long leaseTime, TimeUnit unit) throws InterruptedException {
  // 代码迭代
  //        try {
  //            return tryLockAsync(waitTime, leaseTime, unit).get();
  //        } catch (ExecutionException e) {
  //            throw new IllegalStateException(e);
  //        }
  long newLeaseTime = -1;
  // 判断是否手动设置了释放锁时间
  if (leaseTime != -1) {
    // 当前手动设置了释放锁时间
    // 判断是否手动设置了获取锁的等待时间
    if (waitTime == -1) {
      // 当前没有手动设置获取锁等待时间,获取锁失败不进入重试
      // 释放时间不变
      newLeaseTime = unit.toMillis(leaseTime);
    } else {
      // 当前有手动设置获取锁的等待时间以及释放锁时间
      // 重试可能耗时较久,如果还没有重试完锁就释放了,那么要同时多个节点获取锁还有什么意义的?
      // 将获取锁的等待时间的2倍赋值给newLeaseTime  这样做是为了防止锁过期释放时间leaseTime比等待获取锁时间waitTime小
      // 为什么呢? 如果执行到这一行代码,则表示调用者同时手动设置了leaseTime以及waitTime,设置了leaseTime则不会有watchDog机制进行锁续命,那么在多次获取锁的情况下,并且是多节点同时获取锁成功,一定要保证等待获取锁的过程中,锁一定不能过期被释放,因为redisson的getMultiLock,是通过多个RedissonClient获取的锁,那么可能存在某个线程获取不到锁而等待waitTime,在等待过程中这个锁过期释放了,就不能够保证多个节点同时获取锁
      newLeaseTime = unit.toMillis(waitTime)*2;       
    }
  }

  // 记录当前时间
  long time = System.currentTimeMillis();
  long remainTime = -1;
  if (waitTime != -1) {
    // 如果设置了等待获取锁的时间
    // remainTime 赋值为等待获取锁的时间
    remainTime = unit.toMillis(waitTime); 
  }
  // 计算所等待时间,该方法返回值还是remainTime
  long lockWaitTime = calcLockWaitTime(remainTime);

  // 获取锁失败限制为 0
  int failedLocksLimit = failedLocksLimit(); 
  // 创建集合的大小为locks集合中元素的大小,当前为3
  List<RLock> acquiredLocks = new ArrayList<>(locks.size());
  // 通过for循环遍历每一个lock
  for (ListIterator<RLock> iterator = locks.listIterator(); iterator.hasNext();) {
    --------------------------- 内循环开始-----------------------------
    // 取出lock
    RLock lock = iterator.next();
    boolean lockAcquired;
    try {
      // 没有传递获取锁等待时间、锁释放时间
      if (waitTime == -1 && leaseTime == -1) {
        // 直接获取锁 tryLock(-1, -1, null)
        lockAcquired = lock.tryLock();
      } else {
        // 可能传递了waitTime、leaseTime,或者都传递了
        // 重置获取锁的等待时间,这里lockWaitTime、remainTime是一样的
        long awaitTime = Math.min(lockWaitTime, remainTime);
        // 当前 awaitTime = remainTime = lockWaitTime
        // newLeaseTime 可能为调用者传入的leaseTime,如果调用者没有传入leaseTime,则设置为2被的waitTime
        // 保证newLeaseTime一定大于awatiTime
        lockAcquired = lock.tryLock(awaitTime, newLeaseTime, TimeUnit.MILLISECONDS);
      }
    } catch (RedisResponseTimeoutException e) {
      // 捕获redis响应异常,尝试释放当前可能获取到的锁
      unlockInner(Arrays.asList(lock));
      // 标记当前获取锁失败
      lockAcquired = false;
    } catch (Exception e) {
      lockAcquired = false;
    }

    if (lockAcquired) {
      // 获取锁成功
      // 加入到获取锁成功的集合
      acquiredLocks.add(lock);
    } else {
      // 获取锁失败
      // 应该获取到RedissonLock个数 - 已经获取到RedissonLock个数 是否等允许失败最大锁个数
      // failedLocksLimit 方法返回值默认为0
      // 也就是说for循环遍历到的这个RedissonLock,并且获取锁失败了
      if (locks.size() - acquiredLocks.size() == failedLocksLimit()) {
        // 所有的锁都拿到了才能够结束
        break;
      }

      if (failedLocksLimit == 0) {
        // 把已经获取到的锁释放掉
        unlockInner(acquiredLocks);
        if (waitTime == -1) {
          // 没有获取锁的等待时间,则说明不想重试,直接返回失败
          return false;
        }
        // 想重试
        failedLocksLimit = failedLocksLimit();
        // 把已经拿到的锁释放掉
        acquiredLocks.clear();
        // reset iterator
				// 迭代器指针重置
        while (iterator.hasPrevious()) {
          // 迭代器指针前移
          iterator.previous();
        }
      } else {
        failedLocksLimit--;
      }
    }

    // 判断剩余等待时间
    if (remainTime != -1) {
      // 更新剩余等待时间
      remainTime -= System.currentTimeMillis() - time;
      time = System.currentTimeMillis();
      if (remainTime <= 0) {
        // 说明刚才获取锁已经把等待时间耗尽
        // 把已经获取到的锁释放掉,失败后前面的锁已经不能再拿了,避免其他线程获取锁失败
        unlockInner(acquiredLocks);
        // 返回false获取锁失败
        return false;
      }
      // 如果时间还很充足,则进去下一层循环,继续获取下一把锁
    }
     --------------------------- 内循环结束 -----------------------
  }  // for循环结束

  if (leaseTime != -1) {
    // 手动设置leaseTime才会触发延长锁的过期时间(没有手动设置才会触发watchDog机制)
    // 设置了锁的过期时间
    // 获取所有已经拿到的锁
    List<RFuture<Boolean>> futures = new ArrayList<>(acquiredLocks.size());
    for (RLock rLock : acquiredLocks) {
      // 延长锁的过期时间
      // 因为MutilLock要求获取多节点的锁,在成功获取第一把锁时,锁的过期时间倒计时就已经开始了,这样等多节点获取锁完成之后,获取的第一把锁的过期时间会比最后一把锁的过期时间明显要短
      // 所以等所有锁都拿完了,在为每一把锁延长锁的过期时间
      RFuture<Boolean> future = ((RedissonLock) rLock).expireAsync(unit.toMillis(leaseTime), TimeUnit.MILLISECONDS);
      futures.add(future);
    }

    for (RFuture<Boolean> rFuture : futures) {
      rFuture.syncUninterruptibly();
    }
  }

  return true;
}

5、总结

原理:搭建多个独立的Redis节点,必须在所有节点获取到锁才算真正的获取锁成功
缺陷:运维成本高,实现复杂,如果要保证业务的高可用性,需要搭建多个节点,并且为每个主节点配置从节点,实现主从复制

以上便是搭建Redisson流程以及MutilLock的源码解读,如有误解,请在评论区指出,谢谢

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

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

相关文章

Apache Shiro教程(3)

shiro自定义realms及加密md5salt教程 1、添加pom 文件 <dependency><groupId>org.apache.shiro</groupId><artifactId>shiro-spring</artifactId><version>1.9.1</version> </dependency>2、创建realms自定义文件 import o…

【树莓派/入门】使用MAX30102测量血氧浓度

说在前面 树莓派版本&#xff1a;4b血氧模块&#xff1a;MAX30102树莓派系统&#xff1a;Linux raspberrypi 5.15.76-v8 #1597 SMP aarch64 GNU/Linuxpython版本&#xff1a;3.9.2 模块详情 某宝上买的MAX30102模块&#xff0c;包含杜邦线 准备工作 开启树莓派的GPIO&#x…

java 高级面试题整理

SpringMVC的控制器是单例的吗? 第一次&#xff1a;类是多例&#xff0c;一个普通属性和一个静态属性 总结 尽量不要在controller里面去定义属性&#xff0c;如果在特殊情况需要定义属性的时候&#xff0c;那么就在类上面加上注解Scope("prototype")改为多例的模式…

English Learning - L1-9 时态(中) 2023.1.3 周二

这里写目录标题8 时态8.1 一般时态&#xff08;三&#xff09;一般将来时核心思维&#xff1a;预测&#xff0c;计划&#xff0c;意愿will 和 be going to 的区别将来时的其它表示方式进行时表将来be about to (5 分钟之内)8.2 进行时态核心思维&#xff1a;持续有限的进行&…

黑苹果解决5500xt等navi14显卡引导二阶段黑屏几秒的问题

首先说结论&#xff1a;在注入缓冲帧FB Name的前提下&#xff0c;往显卡注入CFG_LINK_FIXED_MAP参数&#xff0c;类型为Number&#xff0c;值为1。注意一定要注入FB Name&#xff0c;注入FB Name&#xff0c;注入FB Name的前提下&#xff01;&#xff01;我试过不注入FB Name直…

【安全硬件】Chap.5 如何检测芯片中硬件木马?硬件木马的类型有哪些?检测硬件木马的技术

【安全硬件】Chap.5 如何检测芯片中硬件木马&#xff1f;硬件木马的类型有哪些&#xff1f;检测硬件木马的技术前言1. 硬件木马的种类1.1 硬件木马1.2 硬件木马的区分1.1 物理特性类别硬件木马——Physical hardware trojans1.2 激活特性类别硬件木马——Activation1.3 动作特性…

Kafka快速入门

文章目录安装部署集群规划集群部署kafka群起脚本Kafka命令行操作主题命令行操作生产者命令行操消费者命令行操作安装部署 集群规划 集群部署 官方下载地址&#xff1a;http://kafka.apache.org/downloads.html上传安装包到02的/opt/software目录下 [atguiguhadoop02 softwar…

5.hadoop系列之HDFS NN和2NN工作机制

1.第一阶段&#xff1a;NameNode启动 1.第一次启动NameNode格式化后&#xff0c;创建Fsimage和Edits文件&#xff0c;如果不是第一次启动&#xff0c;直接加载Fsimage和Edits到内存 2.客户端对元数据进行增删改请求 3.NameNode记录操作日志&#xff0c;更新滚动日志 4.NameNod…

elasticsearch-head使用问题汇总

1、elasticsearch-head数据预览、基本查询、复合查询模块无法查询es文档记录&#xff08;1&#xff09;解决办法复制vendor.jsdocker cp elasticsearch-head1:/usr/src/app/_site/vendor.js vendor.js修改vendor.js第6886行&#xff0c;将“contentType: "application/x-w…

高性能分布式缓存Redis-第三篇章

高性能分布式缓存Redis-第三篇章一、分布式锁1.1、高并发下单超卖问题1.2、何为分布式锁1.3、分布式锁特点1.4、基于Redis实现分布式锁1.4.1、实现思路&#xff1a;1.4.2、实现代码版本1.4.3、错误解锁问题解决1.4.4、锁续期/锁续命1.4.5、锁的可重入/阻塞锁&#xff08;rediss…

微服务 Spring Boot 整合 Redis BitMap 实现 签到与统计

文章目录⛄引言一、Redis BitMap 基本用法⛅BitMap 基本语法、指令⚡使用 BitMap 完成功能实现二、SpringBoot 整合 Redis 实现签到 功能☁️需求介绍⚡核心源码三、SpringBoot 整合Redis 实现 签到统计功能四、关于使用bitmap来解决缓存穿透的方案⛵小结⛄引言 本文参考黑马 …

【第24天】SQL进阶-查询优化- performance_schema系列实战一:利用等待事件排查MySQL性能问题(SQL 小虚竹)

回城传送–》《32天SQL筑基》 文章目录零、前言一、背景二、performance_schema配置配置表启用等待事件的采集与记录三、sysbench基准测试工具3.1 安装和使用sysbench3.1.1 yum安装3.1.2 查看版本信息3.1.3 sysbench 使用说明3.2 sysbench 测试服务器cpu性能3.3 sysbench测试硬…

Hadoop 入门基础 及HiveQL

一、hadoop 解决了什么问题&#xff1f;即hadoop 产生背景 一个能够轻松方便、经济实惠地存储和分析大量数据的非常流行的开源项目。 二、hadoop 是如何低成本地解决大数据的存储和分析的&#xff1f;即hadoop 原理&#xff0c;hadoop 的组成部分 Hadoop的创始人、Cloudera首…

Java图形化界面---基本组件

目录 一、基本组件介绍 二、Diaolg对话框 &#xff08;1&#xff09;Dialog &#xff08;2) FileDialog 一、基本组件介绍 Button 按钮 Canvas 用于绘图的画布 Checkbox 复选框组件 CheckboxGroup 用于将多个…

【阶段三】Python机器学习06篇:模型评估函数介绍(分类模型)

本篇的思维导图: 模型评估函数介绍(分类模型) accuracy_score()函数 作用:accuracy_score函数计算了模型准确率。在二分类或者多分类中,预测得到的标签,跟真实标签比较,计算准确率。 注意事项:在正负样本不平衡的情况下,准确率这个评价指标有很大的缺陷。比如数据样本…

数据库管理-第五十一期 新年新气象(20230108)

数据库管理 2023-01-08第五十一期 新年新气象1 新年快乐2 旧账3 软硬件对比4 新气象总结第五十一期 新年新气象 1 新年快乐 2023年来了&#xff0c;我也没有第一时间写一篇写文章给大家祝福&#xff0c;第一呢是因为某些原因元旦假期也没咋休息&#xff0c;其次就是因为本周又…

Allegro174版本新功能介绍之新增几种沿着目标打过孔模式

Allegro174版本新功能介绍之新增几种沿着目标打过孔模式 Allegro在低版本的时候,就已经有了沿着目标打过孔的功能,在升级到了174版本后,又新增了几种打过孔的模式,类似下图 以第一种模式举例介绍说明 点击Place

DFT知识点扫盲——DFT scan chain

先说一下tsmc的std celltsmc 7nm工艺下有专门的std synccell 命名如下&#xff1a;SDFSYNC1RPQD1XXXXVTSDFSYNC1SNQD1XXXXVTSDFSYNC1QD1XXXXVT不考虑VT, PWR和track&#xff0c;电压等差别&#xff0c;整个工艺库下只有这三种实际在项目中synccell一般直接上ULVT&#xff0c;既…

2022年第四届全国高校计算机能力挑战赛c++组决赛

A 题目描述 小丽好朋友的生日快到了&#xff0c;她打算做一些折纸放在幸运罐中作为生日礼物。小丽计划总共 需要a颗星星以及b只纸鹤。现在市场上卖的到的星星纸(折小星星的专用纸)一张可以折c颗小星星&#xff0c;一张纸鹤纸(折纸鹤的专用纸)可以折d只小纸鹤。她准备一共买k张…

【C++】模板进阶

​&#x1f320; 作者&#xff1a;阿亮joy. &#x1f386;专栏&#xff1a;《吃透西嘎嘎》 &#x1f387; 座右铭&#xff1a;每个优秀的人都有一段沉默的时光&#xff0c;那段时光是付出了很多努力却得不到结果的日子&#xff0c;我们把它叫做扎根 目录&#x1f449;非类型模…