【Redis】分布式锁及其他常见问题

news2025/1/14 18:16:10

分布式锁及其他常见问题

1. 我看你的项目都用到了 Redis,你在最近的项目的哪些场景下用到了 Redis 呢?

一定要结合业务场景来回答问题!要是没有不要硬讲,除非面试官问;

在这里插入图片描述

接下来面试官将深入发问。

  • 你没用到的也可能会试探着去问;

Redis 分布式锁使用的场景?

分布式情况下的,或者集群情况下的,多个微服务操作同一对象,可能是相同操作(同时改),也可能是不同操作(一个删,一个改…)

  1. 定时任务
  2. 抢单/秒杀
  3. 密等性场景

2. 抢卷场景

2.1 分布式问题

在这里插入图片描述

执行流程:

在这里插入图片描述

而如果是分布式的项目,就可能对服务进行集群部署:

在这里插入图片描述

2.2 synchronized 锁解决问题

在这里插入图片描述

显然这解决不了分布式问题;

只能解决同个 Java 进程的多线程的问题,也就是处理同一个 Java 服务的多个请求的问题;

在这里插入图片描述

2.3 分布式锁解决问题

在这里插入图片描述

3. 分布式锁的实现原理

Redis 分布式锁其实就是“争夺一个 key 的定义”,主要利用 Redis 的 setnx 命令(SET if **not ex**ists 如果不存在,则设置)

在这里插入图片描述

在这里插入图片描述

在这里插入图片描述

小插曲:锁提前过期,但是业务未结束,锁岂不是可以被其他服务获取了?

其实这个问题可以通过给锁续期来解决,举个例子:

一个分布式锁有效时长是 5 秒,但是业务时长 7 秒,我们可以每隔一段时间,如每个 3 秒 就判断业务是否结束,如果结束了那就释放锁,如果没结束,就重置锁的有效时长,如重置为 5 秒;

4. 程序中怎么使用分布式锁?

分布式锁真的很灵活和精准!

可以使用 redission 框架(加锁、设置过期时间等操作都是基于 lua 脚本,Redis可以识别的可保证原子性的脚本完成),执行流程如下:

在这里插入图片描述

示例代码:

  • R 在 redisson 中值得就是 Redis 的 R;

在这里插入图片描述

RLock 锁是可以重入的锁

在这里插入图片描述

5. 分布式锁主从一致性的问题

在这里插入图片描述

正常情况下,Redis 集群的主从会进行同步,所以不同的 Java 应用访问 Redis 集群的时候,是只有一个应用能获取一把锁;

但是,如果主节点在从节点同步之前挂了,从节点就变成了新的主节点,另一个 Java 应用尝试获取这把锁,发现可以获取成功,就出现了两个线程同时持有一把锁的情况了;

之后,两个线程就没有阻塞的同时进行了。

主从就不一致了,因为从节点只显示正常一个线程获得锁,但实际上是两个线程用了一把锁;

在这里插入图片描述

可以使用分布式锁的另一种实现:RedLock(**Red**is)

RedLock(红锁),在多个 Redis 实例上创建锁 (n / 2 + 1),避免在一个 Redis 实例上加锁:

在这里插入图片描述

缺点:

  1. 实现复杂
  2. 性能差
  3. 运维繁琐

如果非要解决这个问题,可以用 zookeeper(CP 思想)去解决,而 Redis (AP 思想)更适合根据具体业务实现最终一致性;

Redis 分布式锁,是如何实现的?

  • 根据自己简历的项目进行描述分布式锁使用后的场景,如抢票、维持幂等性…
  • 我们当时使用的是 redisson 实现的分布式锁,底层是 setnx 和 lua 脚本(保证原子性)

Redisson 实现的分布式锁是如何合理的控制锁的有效时长的?

  • 在 redisson 的分布式锁中,提供了一个 ==WatchDog(看门狗)==一个线程获得锁成功以后 WatchDog 会给持有锁的线程**续期(默认是每个 10 秒续期一次)**

Redisson 的这个锁,是可重入锁么?

  • 可以重入,多个锁重入需要判断是否是当前线程,在 Redis 中进行存储的时候使用 hash 结构,来存储==线程信息和重入的次数==;

Redisson 的这个锁,能处理主从数据一致的问题吗?

  • 主从数据一致的问题:主节点在从节点同步之前宕机,从节点就被升级为主节点的数据不一致的问题;

  • 处理不了,但是可以用 Redisson 提供的的==红锁来解决,但是这样的话,性能太低了==,如果是 AP,则为我们保证数据的最终一致性即可,如果业务要求 CP,建议采用 zookeeper 实现的分布式锁;

6. Redis 集群有哪些方案?

在 Redis 中提供了三种集群方案:

  1. 主从复制
  2. 哨兵模式
  3. 分片集群

有一些常见的问题:

  1. Redis 主从数据同步的流程是什么?
  2. 怎么保证 Redis 的高并发高可用?
  3. 你们是用 Redis 的单点还是集群,哪种集群?
  4. Redis 分片集群中数据是怎么存储和读取的?
  5. Redis 集群脑裂,该怎么解决呢?

6.1 主从复制

主从同步原理:

由于单点 Redis 的并发能力是有上限的,要进一步提高 Redis 的并发能力,就需要搭建主从集群,实现读写分离;

在这里插入图片描述

主从全量同步:

在这里插入图片描述

主从增量同步(slave 重启或后期数据变化)

在这里插入图片描述

主从同步数据的流程:

  1. 从节点请求主节点同步数据(replication id、offset)
    • replication id 用于判断是否是对应的主节点;
  2. 主节点判断是否是第一次请求(第一次就是全量同步),发送给从节点同步版本需要的信息(replication id、offset)
  3. 执行 bgsave,生成 rdb 文件后,发送给从节点去执行;
  4. 从节点清空数据,加载 rdb 文件(全量同步加载时间可能较长),期间主节点的新增数据以命令的方式记录在缓冲区(一个日志文件)
  5. 把生成之后的命令日志文件发送给从节点进行同步;
  • 之后从节点重启或者日常的同步数据:
  1. 请求主节点同步数据(replication id、offset)
  2. 主节点判断是否是第一次请求(不是第一次就是增量同步),根据 offset 获取 offset 之后的命令文件,发送给从节点;
  3. 从节点执行命令进行数据同步;

6.2 哨兵模式

在这里插入图片描述

Redis 提供了哨兵(Sentinel)机制来实现主从集群的自动故障恢复。哨兵的结构和作用如下:

  1. 监控:Sentinel 会不断检查您的主从系欸但是否按预取工作;
  2. 自动故障恢复:如果主节点故障,Sentinel 会选取一个从节点为新的主节点,故障实例恢复后作为从节点以新的主节点为主;
  3. 通知:Sentinel 充当 Redis 客户端的服务发现来源,当集群发生故障转移的时候,会将最新的信息推送给 Redis 客户端,也就是说之后的客户端能正确拉取 Redis 服务;

状态监控的相关细节:

  • Sentinel 基于心跳机制检测服务状态,每隔一秒向集群的每个实例发送 ping 命令;
  • 主观下线:如果某个 Sentinel 节点发现某实例未在规定时间响应,则认为该实例主观下线;
    • Sentinel 有集群,但是没有主从之分,没有“Sentinel 监控 Sentinel 的套娃”🤣
  • 客观下线:若超过指定数量(quorum)的 Sentinel 主观认为该实例下线,则该实例客观下线;
    • quorum 最好超过 Sentinel 实例数量的一半;

在这里插入图片描述

哨兵选主规则:

  1. 首先,判断主从节点断开时间长短,如果超过指定值就排除该从节点(主节点都挂了,从节点断开连接太慢,太迟钝的节点直接不要);
  2. 其次,判断从节点的 slave-priority 值,越小优先级越高;
  3. 如果 slave-priority 相等,则判断 offset 值,越大优先级越高(越大代表数据同步率高);
    • 大部分情况下是通过这个判断出来的;
  4. 最后,判断从节点的运行 id 大小,越小优先级越高;

脑裂:

正常情况下:

在这里插入图片描述

由于网络原因,主节点与哨兵在不同的网络分区,哨兵不再监控主节点,那哨兵就会觉得主节点挂了,重新选择一个主节点,那就同时存在两个主节点了!(也就是 Redis 的主节点大脑分裂了)

客户端连接的是旧的主节点,客户端会一直写入旧的主节点,因为旧的主节点没有从节点,期间客户端读的是旧的主节点,而新的主节点的从节点一直没有同步数据;

网络恢复前,旧的主节点也不知道新的主节点那边的数据;

在这里插入图片描述

网络恢复后,会将旧的主节点强制转为从节点,会从新的主节点同步数据,那么期间旧的主节点的新增数据将直接丢失;

在这里插入图片描述

解决/缓解的方案:

  1. min-replicas-to-write 1 表示最少的从节点为一个
    • 如果主节点没有从节点,不可信;
  2. min-replicas-max-lag 5 表示数据复制和同步的延迟不能超过 5 秒,否则拒绝请求;
    • 也就是说,本次请求距离上次同步,不能超过 5 秒;

怎么保证 Redis 高并发下的高可用:

  • 哨兵模式:实现主从集群的自动故障恢复(监控、自动故障恢复、通知)

规模不算很大的项目,用主从 + 哨兵就够了:一主一从一哨兵,顺便预防一下脑裂,单节点不超过 10 G 内存,如果 Redis 内存不足则可以给不同服务分配独立的 Redis 主从节点;

  • 也就是一个需要很大的服务,单独去使用一个 Redis 主从集群,不与其他服务共用;
  • 不同项目则一般用不同的 Redis,避免 key 冲突,当然同一个项目的服务也会出现冲突的问题,但是同一个项目较容易的去控制和管理;

Redis 集群脑裂的现象如何解决?

  • 我们可以修改 Redis 的配置,可以设置最少的从节点数、缩短主从数据同步的延迟时间,达不到要求就拒绝请求,这样可以避免 Redis 大量数据的丢失;

6.3 分片执行

主从+哨兵可以解决高可用的高并发读的问题,但是依旧存在两个问题:

  1. 海量数据存储问题
  2. 高并发写的问题

使用分片集群可以解决上述问题,分片集群特征:

  1. 集群中有多个主节点,每个主节点保存不同数据;
  2. 每个主节点,可以有多个从节点(主从复制);
  3. 主节点之间相互通过心跳,检测彼此的健康状态(不需要通过哨兵);
  4. 客户端请求可以访问集群的容易节点,最终都会被转发给正确的节点;

在这里插入图片描述

数据读写:

Redis 分片集群引入了哈希槽的概念,Redis 集群有 16384 个哈希槽,每个 key 通过 CRC 16 校验后对 16384 取模,取模的结果就是插槽,找到插槽对应的哈希槽实例,集群的每个节点负责一部分哈希槽;

  • 读操作用同样的方式寻找 key 的插槽并找到对应的值;

补充:根据的是 key 的有效部分,如果 key 前面有大括号,大括号包裹的就是有效部分,否则 key 本身就是有效部分;还有就是,这个只是确定在哪个哈希槽,最终获取 key 的值还是要通过 key 本身,也就是获取值的时候还是要通过 {aaa}key;

在这里插入图片描述

7. Redis 是单线程的,但是为什么还是那么快

在这里插入图片描述

  • Redis 是纯内存操作,执行速度非常快;
  • 采用单线程,避免不必要的上下文切换,多线程还要考虑多线程安全问题,比如使用各种锁;

如果 Redis 底层用到锁,与用 Redis 分布式锁是不一样的;

Redis 分布式锁并不是解决 Redis 的线程安全问题,而是解决业务的线程安全问题,因为多次发送给 Redis 的操作不能保证原子性的,多线程的话可能会导致 Redis 操作的顺序里穿插了别的线程的 Redis 操作;

所以要用 Redis 事务保证一串 Redis 写操作的原子性:

redisTemplate.execute(new SessionCallback() {
   @Override
   public Object execute(RedisOperations redisOperations) throws DataAccessException {
       redisOperations.multi();
       // runnable 方法中的 Redis 操作,得用这里同一个 redisTemplate 去操作 Redis 的代码才能被放入事务块
       runnable.run();
       return redisOperations.exec();
   }
});

当然,如果是抢票之类的场景,还是要用 Redis 分布式锁;

  • Redis 使用 I/O 多路复用模型,非阻塞 IO;

8. I/O 多路复用模型(了解)

在这里插入图片描述
在这里插入图片描述

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

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

相关文章

VS Code 配置 cmake

手动添加 CMake 编译器的搜索路径 如果没有设置上面的路径,有些编译器是找不到的

基于SpringBoot和Vue的校园周边美食探索以及分享系统

今天要和大家聊的是基于SpringBoot和Vue的校园周边美食探索以及分享系统 !!! 有需要的小伙伴可以通过文章末尾名片咨询我哦!!! 💕💕作者:李同学 💕&#x1f…

kali 渗透工具 - mestaploit

永恒之蓝漏洞的小知识: 黑客通过改造 永恒之蓝 制作 wannacry 制作病毒入侵高校内网。 mestaploit 攻击永恒之蓝流程: 使用模块 msfconsole配置required 模块参数运行,开始监听主机 msfconsole 主要模块 - 选择使用模块 search ms17_01…

工业组态 物联网组态 组态编辑器 web组态 组态插件 编辑器

体验地址:by组态[web组态插件] BY组态是一款非常优秀的纯前端的【web组态插件工具】,可无缝嵌入到vue项目,react项目等,由于是原生js开发,对于前端的集成没有框架的限制。同时由于BY组态只是一个插件,不能独…

大转盘抽奖小程序源码

源码介绍 大转盘抽奖小程序源码,测试依旧可用,无BUG,跑马灯旋转效果,非常酷炫。 小程序核心代码参考 //index.js //获取应用实例 var app getApp() Page({data: {circleList: [],//圆点数组awardList: [],//奖品数组colorCirc…

SCI一区 | Matlab实现NGO-TCN-BiGRU-Attention北方苍鹰算法优化时间卷积双向门控循环单元融合注意力机制多变量时间序列预测

SCI一区 | Matlab实现NGO-TCN-BiGRU-Attention北方苍鹰算法优化时间卷积双向门控循环单元融合注意力机制多变量时间序列预测 目录 SCI一区 | Matlab实现NGO-TCN-BiGRU-Attention北方苍鹰算法优化时间卷积双向门控循环单元融合注意力机制多变量时间序列预测预测效果基本介绍模型…

设计模式总结-面向对象设计原则

面向对象设计原则 面向对象设计原则简介单一职责原则单一职责原则定义单一职责原则分析单一职责原则实例 开闭原则开闭原则定义开闭原则分析开闭原则实例 里氏代换原则里氏代换原则定义里氏代换原则分析 依赖倒转原则依赖倒转原则定义依赖倒转原则分析依赖倒转原则实例 接口隔离…

配置 施耐德 modbusTCP 分布式IO子站 RPA0100

1. 总体步骤 2. 软件组态:在 Unity Pro 软件中创建编辑 PRA 模块工程 2.1 新建项目 模块箱硬件型号如下 点击 Unity Pro 软件左上方【新建】按钮,选择正确的 DIO 模块型号、背板型号 2.2 模块组态 2.2.1 拖拽添加模块 双击【配置】菜单下的【0&…

糟糕,Oracle归档满RMAN进不去,CPU98%了!

📢📢📢📣📣📣 哈喽!大家好,我是【IT邦德】,江湖人称jeames007,10余年DBA及大数据工作经验 一位上进心十足的【大数据领域博主】!😜&am…

【QT+QGIS跨平台编译】056:【pdal_lepcc+Qt跨平台编译】(一套代码、一套框架,跨平台编译)

点击查看专栏目录 文章目录 一、pdal_lepcc介绍二、pdal下载三、文件分析四、pro文件五、编译实践一、pdal_lepcc介绍 pdal_lepcc 是 PDAL(Point Data Abstraction Library)的一个插件,用于点云数据的压缩。它基于 EPCC(Entwine Point Cloud Compression)算法,提供了对点…

72小时内报告!美国发布关键基础设施网络攻击通报新规草案

美国网络安全和基础设施安全局(CISA)本周四发布了关键基础设施企业如何向政府报告网络攻击的规定草案。 新规基于拜登2022年3月15日签署的美国《关键基础设施网络事件报告法案》(简称CIRCIA)。这是美国联邦政府首次提出一套跨关键基础设施部门的全面网络安全规则。CISA正在就规…

国外服务器托管需要了解哪些信息

国外服务器托管服务提供了一种在国外租用并管理服务器的方式,适用于需要特定地域服务或对本地法规有特殊要求的企业和个人。那么想要进行国外服务器托管需要了解哪些信息呢?Rak部落小编为您整理发布国外服务器托管相关内容。 以下是一些关于国外服务器托管服务的详…

十分钟掌握在 PyTorch 中构建一个深度神经网络,基本组件、步骤和代码实现,从导入模块和定义网络结构到训练和评估网络性能。

🍉 CSDN 叶庭云:https://yetingyun.blog.csdn.net/ 深度神经网络(Deep Neural Networks, DNNs),也被称为人工神经网络(Artificial Neural Networks,ANNs),已成为当今机器学习任务中最流行、最成功的方法之一。这些网络能够表示数据中的复杂关系,并在图像分类、自然…

储能系统--液冷充电枪

前言 随着新能源汽车在市场中的占比不断攀升,续航里程和充电时间成为了制约新能源汽车发展的两个关键因素, 而随着续航里程的增加,电池容量也会相应的增加,充电时间也会加长,大功率快充技术逐渐成为解决续航瓶颈的关键…

C易错注意之const修饰指针,含char类型计算,位段及相关经典易错例题

目录 前言 一:const修饰指针 1.const修饰变量 2.const 修饰指针 1.const int*p&m; 2. int* const p&m; 3. int const *p&m; 4. int const *const p&m; 5.总结 总之一句话为:左定值有定向 二:关于计算中char类型…

AI工作站设计方案:903-多路PCIe3.0的单CPU 学习型AI工作站

多路PCIe3.0的单CPU 学习型AI工作站 一、机箱功能和技术指标: 系统 系统型号 ORI-SR500 主板支持 EEB(12*13)/CEB(12*10.5)/ATX(12*9.6)/Mi cro ATX 前置硬盘 最大支持2个3.5寸1个2.5寸SATA 硬盘2个2.5寸SATA 硬盘 (背部) 电源类型 C…

Mybatis--TypeHandler使用手册

TypeHandler使用手册 场景:想保存user时 teacher自动转String ,不想每次保存都要手动去转String;从DB查询出来时,也要自动帮我们转换成Java对象 Teacher Data public class User {private Integer id;private String name;priva…

gpt国内怎么用?最新版本来了

claude 3 opus面世后,这几天已经有许多应用,而其精确以及从不偷懒(截止到2024年3月11日还没有偷懒)的个性,也使得我们可以用它来首次完成各种需要多轮对话的尝试。 今天我们想要进行的一项尝试就是—— 如何从一个不知…

PWM技术的应用

目录 PWM技术简介 PWM重要参数 PWM实现呼吸灯 脉宽调制波形 PWM案例 电路图 keil文件 直流电机 直流电机的控制 直流电机的驱动芯片L293D L293D引脚图 L293D功能表 直流电机案例 电路图 keil文件 步进电机 步进电机特点 步进电机驱动芯片L298 L298引脚图 L…

MySQL、Oracle查看字节和字符长度个数的函数

目录 0. 总结1. MySQL1.1. 造数据1.2. 查看字符/字节个数 2. Oracle2.1. 造数据2.2. 查看字符/字节个数 0. 总结 databasecharbyteMySQLchar_length()length()Oraclelength()lengthB() 1. MySQL 1.1. 造数据 sql drop table if exists demo; create table demo (id …