Redis核心技术与实战【学习笔记】 - 9.如何避免单线程模型的阻塞

news2025/1/10 17:28:38

概述

Redis 被广泛应用的原因是因为它支持高性能访问。所以,我们要重视所有可能影响 Redis 性能的因素(如命令操作、系统配置、关键机制、硬件配置等)。

影响 Redis 性能的 5 大方面的潜在因素分别是:

  • Redis 内部的阻塞式操作
  • CPU 核和 NUMA 架构的影响
  • Redis 关键系统配置
  • Redis 内存碎片
  • Redis 缓冲区

1. Redis 实例有哪些阻塞点?

Redis 实例要和许多对象进行交互,这些不同的交互就会涉及不同的操作,我们来看看客户端和 Redis 实例交互的对象,以及交互时会发送的操作。

  • 客户端:网络 IO,键值对增删改查操作,数据库操作
  • 磁盘: 生成 RDB 快照,记录 AOF 日志,AOF 日志重写。
  • 主从节点:主库生成、传输 RDB 文件,从库接收 RDB 文件、清空数据库、加载 RDB 文件;
  • 切片集群实例:向其他实例传输哈希槽信息,数据迁移。
    在这里插入图片描述

1.1 和客户端交互的阻塞点

网络 IO 有时候会比较慢,但是 Redis 使用了 IO 多路复用机制,避免了主线程一直处在等待网络连接或者请求到来的状态,所以网络 IO 不是导致 Redis 阻塞的因素。

键值对的增删改查操作是 Redis 和 客户端交互的主要部分,也是 Redis 主线程执行的主要任务。所以,复杂度高的增删改查操作会阻塞 Redis。

判断操作复杂度有一个最基本的标准,就是看操作的复杂度是否为 O(N).

Redis 中涉及集合的操作复杂度通常为 O(N),所以我们要重视起来。例如集合元素全量查询 HGETALL、SMEMBERS,以及集合的统计聚合操作,例如求交、并和差集。这些操作可以作为 Redis 的第一个阻塞点:集合的全量查询和聚合操作

另外,集合自身的删除操作,同样也有潜在的阻塞风险。这是因为删除不仅仅只是把数据删除,还要释放键值对占用的内存空间。

你可以不要小瞧内存释放的过程。释放内存只是第一步,为了高效管理内存空间,在应用程序释放时,操作系统需要把释放掉的内存块插入一个空间内存块的链表,以便后续进行管理和再分配。这个过程本身需要一定的时间,而且会阻塞当前释放内存的应用程序,所以,如果一下子释放了大量的内存,空闲内存块链表的操作时间就会增加,相应地就会造成 Redis 主线程的阻塞。

什么时候会释放大量内存呢? 其实就是在删除大量键值对数据的时候,最典型的就是删除了包含了大量元素的集合,也称为 bigkey 删除。下图是测试了不同元素数据的集合在进行删除操作时所消耗的时间:
在这里插入图片描述
根据压测,可以得出结论:

  1. 当数据量从10 万增加到 100 万时,4 大集合类型的删除时间的增长幅度从5倍上升到了近 20 倍。
  2. 集合元素越大,删除所花费的时间越长。
  3. 当删除有 100 个元素的集合时,最大删除时间已经达到了 1.9 秒(Hash 类型)。Redis 的响应时间一般在微妙级别,所以,一个操作达到近 2 秒,不可避免地会阻塞主线程。

很显然,Redis 的第二个阻塞点:bigkey 删除操作

删除操作对 Redis 实例性能的负面影响很大,而且在实际业务开发时很容易被忽略,所以移动要重视它

既然频繁你删除键值对都是潜在的阻塞点,那么,Redis 的数据库级别操作中,清空数据库(如 FLUSHDB 和 FLUSHALL 操作)必然也是一个潜在的阻塞风险,因为它设计到删除和释放所有的键值对。所以,Redis的第三个阻塞点:清空数据库

1.2 和磁盘交互时的阻塞点

磁盘 IO 一般都是比较耗时费力的,需要重点关注。不过,Redis 开发者早就认识到磁盘 IO 会带来阻塞,所以就把 Redis 设计为采用子进程的方式生成 RDB 快照文件,以及执行 AOF 日志重写操作。这样一来,这两个操作由子进程负责执行,慢速的磁盘 IO 就不会阻塞主线了。

但是,Redis 直接记录 AOF 日志时,会根据不同的写回策略对数据做落盘保存。一个 AOF 同步写磁盘的操作的耗时大约是 1~2 ms,如果有大量的写操作需要记录在 AOF 日志中,并同步写回的话,就会阻塞主线程了。这就是 Redis 的第四个阻塞点: AOF 日志荣同步写

1.3 主从节点交互时的阻塞点

主从集群中,主库需要生成 RDB 文件,并传输给从库。主库在复制过程中,创建和传输 RDB 文件都是由子进程来完成,不会阻塞主线程。但是对于从库来说,它在接收了 RDB 文件后,需要使用 LPUSHDB 命令清空当前数据库,这就正好是刚刚分析的第三个阻塞点

此外,从库在情况当前数据库后,还需要把 RDB 文件加载到内存,这个过程的快慢和 RDB 文件的大小密切相关, RDB 文件越大,加载过程越慢,所以 加载 RDB 文件就称为了 Redis 的第五个阻塞点

1.4 切片集群实例交互时的阻塞点

当我们部署 Redis 切片集群时,每个 Redis 实例上分配的哈希槽信息需要在不同实例间进行传输,同时,当需要进行负载均衡或者实例增删时,数据会在不同的实例间进行迁移。不过哈希槽信息量不大,而数据迁移是渐进式执行的,所以一般来说,这两类操作对 Redis 主线程的阻塞风险不大。

不过,如果你使用了 Redis Cluster 方案,而且同时正好迁移的是 bigkey 的话,就会造成主线程的阻塞,因为 Redis Cluster 使用了同步迁移。所以,你要知道,当没有 bigkey 时,切片集群的各实例在进行交互时不会阻塞主线程。

1.5 小结

好了,经过上面分析 Redis 的各种关键操作,总结下刚刚找到的阻塞点:

  • 集合全量查询和聚合操作
  • bigkey 删除
  • 清空数据库
  • AOF 日志同步写
  • 从库加载 RDB 文件

如果在主线程执行这些操作,必然会导致主线程的阻塞。为了避免阻塞式操作,Redis 提供了异步线程机制。所谓异步线程机制,就是指 Redis 会启动一些子线程,然后把一些任务交给这些子线程,让它们在后台完成,而不再由主线程来执行这些任务。使用异步线程机制执行操作,可以避免阻塞主线程。

不过,这个时候,问题来了:这五大阻塞式操作都可以被异步执行吗?

2.哪些阻塞点可以异步执行?

如果一个操作能被异步执行,就意味着它不是 Redis 主线程的关键路径上的操作。关键路径上的操作就是,客户端把请求发给 Redis 后,等着 Redis 返回数据结构的操作。

在这里插入图片描述

  • 主线程收到操作 1 后,因为操作 1 并不用给客户端返回具体数据,所以,主线程可以把它交给后台子线程来完成,同时只要给客户端返回一个 “OK” 结果就行。
  • 在子线程执行操作 1 的时候,客户端又向 Redis 实例发送了操作 2,而此时客户端需要使用操作 2 返回的数据结果的,如果操作 2 不返回结果,那么客户端将一直处于等待状态。

在这个例子中,操作 1 就不算关键路径上的操作,所以可以有后台子线程异步执行。而操作 2 需要把结果返回给客户端,它就是关键路径上的操作,所以主线程立即把这个操作执行完。

对于 Redis 来说,读操作是典型的关键路径操作,因为客户端发送了读操作之后,就会等待读取数据的返回。而 Redis 的第一个阻塞点 “集合全量查询和聚合操作” 都涉及到了读操作,所以它是不能进行异步操作了。

删除操作,并不需要给客户端返回具体的数据结果,所以不算关键路径操作。而我们刚才总结的 第二个阻塞点 “bigkey 删除”,和第三个阻塞点 “清空数据库” 都是对数据做删除,并不在关键路径上。因此可以使用后台子线程来异步执行删除操作

最后,我们看下“从库加载 RDB 文件”这个阻塞点。从库想要对客户端提供数据存取服务,就必须把 RDB 文件加载完成,所以,这个操作也属于关键路径上的操作,必须让从库的主线程来执行。

3.异步的子线程机制

Redis 主线程启动后,会使用操作系统提供的 pthread_create 函数创建 3 个子线程,分别由他们负责 AOF 日志写操作、键值对删除以及文件关闭的异步执行。

主线程通过一个链表形式的任务队列和子线程进行交互。当收到键值对删除和清空数据库的操作时,主线程会把这个操作封装成一个任务,放入任务队列中,然后给客户端返回一个完成信息,表明删除已经完成。

但实际上,这个时候删除还没执行,等到后台子线程从任务队列中读取数据,才实际删除键值对,并释放响应的内存空间。因为,我们把这种异步删除也称为惰性删除(lazy free)。此时,删除或清空操作不会阻塞主线程,这就避免了对主线程的性能影响。

和惰性删除类似,当 AOF 日志配置成 everysec 选项后,主线程会把 AOF 写日志操作封装成一个任务,也放到任务队列中。后台子线程读取任务后,开始自行写入 AOF 日志,这样主线程就不用一直等待 AOF 日志写完了。

下图展示了 Redis 中异步子线程执行机制。
在这里插入图片描述
不过,有个地方需要注意下,异步键值对删除和情况数据库操作是 Redis 4.0 后提供的功能,Redis 也提供了新的命令来执行这两个操作。

  • 异步键值对删除:当你的集合类型中有大量元素(如有百万级别或者千万级别元素)需要删除时,建议你使用 ULINK 命令
  • 清空数据库:可以在 FLUSHDB 和 FLUSHALL 命令后机上 AYSNC 选项,这样可以让后台子线程异步地清空数据库
FLUSHDB AYSNC
FLUSHALL AYSNC

4.小结

本章,我们学习了 Redis 实例允许时的 4 大类交互对象:客户端、磁盘、主从库实例、切片集群实例。基于这 4 大类交互对象,我们梳理了会导致 Redis 性能受损的 5 大阻塞点,包括集合全量查询和聚合操作、bigkey 删除、清空数据库、AOF 日志同步写,以及从库加载 RDB 文件。

在这 5 大阻塞点中,bigkey 删除、情况数据库、AOF 日志同步写不属于关键路径操作,可以使用异步子线程机制来完成。 Redis 在运行时会创建三个子线程,主线程会通过一个任务队列和三个子线程进行交互。子线程会根据任务的具体类型,来执行响应的异步操作。

不过 异步删除操作是 Redis 4.0 以后才有的功能,如果你使用的是 4.0 之前的版本,当你遇到 bigkey 删除时,给你个小建议先使用集合类型提供的 SCAN 命令读取数据,然后再次进行删除。因为 SCAN 命令可以每次只读取一部分数据并删除,这样可以避免一次性删除大量 key 给主线程带来阻塞。
例如,对于 Hash 类型的 bigkey 删除,你可以使用 HSCAN 命令,每次从 Hash 集合中获取一部分键值对(如 200 个),再使用 HDEL 删除这些键值对,这样就可以把删除压力分摊到多次操作中,那么,每次删除操作的耗时就不会太长,也就不会阻塞主线程了。

最后,集合全量查询和聚合操作、从库加载 RDB 文件是在关键路径上,无法使用异步操作来完成,对于这两个阻塞点的小建议:

  • 集合全量查询和聚合操作可以使用 SCAN 命令,分批读取数据,再在客户端进行聚合计算
  • 从库加载 RDB 文件把主库的数据量大小控制在 2~4GB 左右,以保证 RDB 文件能以较快的速度加载。

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

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

相关文章

leetcode209长度最小的子数组|滑动窗口算法详细讲解学习

滑动窗口是一种基于双指针的一种思想,两个指针指向的元素之间形成一个窗口。 分类:窗口有两类,一种是固定大小类的窗口,一类是大小动态变化的窗口。 简而言之,滑动窗口算法在一个特定大小的字符串或数组上进行操作&…

git clone常见问题一览及解决方法

在使用Ubuntu下,终端运行git clone命令时会遇见许多问题,本文主要针对一些常见的问题进行整理。关于换源问题,推荐使用小鱼的一键换源。 1.git clone 速度过慢 1.1 魔法 这个方法不做过多赘述,ubuntu下个人使用发现体验良好&am…

Websocket基本用法

1.Websocket介绍 WebSocket是基于TCP的一种新的网络协议。它实现了浏览器与服务器全双工通信——浏览器和服务器只需要完成一次握手,两者之间就可以创建持久性的连接,并进行双向数据传输。 应用场景: 视频弹幕网页聊天体育实况更新股票基金…

BUUCTF-Real-[PHP]XXE

目录 1、原理 2、XXE漏洞产生的原因 3、开始复现 paylaod 复现 4、flag 1、原理 XML数据在传输过程中,攻击者强制XML解析器去访问攻击者指定的资源内容(本地/远程),外部实体声明关键字SYSTEM会令XML解析器读取数据&#xf…

八斗学习笔记

1 初始环境安装 Anaconda安装(一款可以同时创建跟管理多个python环境的软件) https://blog.csdn.net/run_success/article/details/134656460 Anaconda创建一个新python环境(安装人工智能常用的第三方python包,如:tensorflow、keras、pytorch) https://…

机器学习4-多元线性回归

多元线性回归(Multiple Linear Regression)是线性回归的一种扩展形式,用于建立因变量与多个自变量之间的关系。在简单线性回归中,我们考虑一个因变量和一个自变量之间的线性关系,而多元线性回归允许我们考虑多个自变量对因变量的影响。 一般…

MYSQL的配置和安装

下载安装 打开官网 MYSQL官网 点击DOWNLOADS 滑到最低下点击:MYSQL Community(GPL) Downlads 点击Download Archives 点击MySQL Community Server进入网站 选择相应版本下载,这里选择的是5.7.24版本,x86 64位【按需选择】 下载解压 配置文件…

微服务中间件 RabbitMq学习

1、为什么需要Mq 例如在用户注册业务中,用户注册成功后 需要发注册邮件和注册短信,传统的做法有两种 1.串行的方式;2.并行的方式 ; 假设三个业务节点分别使用50ms,串行方式使用时间150ms,并行使用时间10…

k8s从私有库harbor中拉取镜像

一、前言 Docker镜像是构建应用程序的基础。然而,许多组织和开发团队希望保留他们的Docker镜像在私有仓库中,并从中拉取镜像,而不是从公共Docker Hub中下载。这样做的原因有很多,包括: 1. 安全性:私有仓库可…

Vue学习笔记(二)快速入门

Vue学习笔记&#xff08;二&#xff09;快速入门 vue小试牛刀 hello-vue3.html <body><div id"app"><h1>{{msg}}</h1></div><script type"module">import {createApp} from https://unpkg.com/vue3/dist/vue.esm-b…

电脑用的视频编辑软件有哪些 视频剪辑软件排行榜 视频剪辑软件推荐 视频剪辑培训学习 视频剪辑制作自学 电脑视频剪辑需要什么配置

电脑视频剪辑软件这么多&#xff0c;到底哪些比较好用&#xff1f;下面就让我们以十大电脑视频剪辑软件排行榜来细数好用的软件。另外&#xff0c;电脑视频剪辑需要什么配置&#xff1f;本文也会给大家从内存、CPU等参数上介绍&#xff0c;并推荐好用的电脑设备。 一、十大电脑…

代码随想录算法训练营第36天 | 435.无重叠区间 + 763.划分字母区间 + 56.合并区间

今日任务 435. 无重叠区间 763.划分字母区间 56. 合并区间 435.无重叠区间 - Medium 题目链接&#xff1a;力扣&#xff08;LeetCode&#xff09;官网 - 全球极客挚爱的技术成长平台 给定一个区间的集合 intervals &#xff0c;其中 intervals[i] [starti, endi] 。返回 需…

【Linux】多线程(线程概念+线程控制)

&#x1f307;个人主页&#xff1a;平凡的小苏 &#x1f4da;学习格言&#xff1a;命运给你一个低的起点&#xff0c;是想看你精彩的翻盘&#xff0c;而不是让你自甘堕落&#xff0c;脚下的路虽然难走&#xff0c;但我还能走&#xff0c;比起向阳而生&#xff0c;我更想尝试逆风…

vue3 [Vue warn]: Unhandled error during execution of scheduler flush

文章目录 前言一、报错截图二、排除问题思路相关问题 Vue3 优雅解决方法异步组件异同之处&#xff1a;好处&#xff1a;在使用异步组件时&#xff0c;有几个注意点&#xff1a; vue3 定义与使用异步组件 总结 前言 Bug 记录。开发环境运行正常&#xff0c;构建后时不时触发下面…

5、应急响应-拒绝服务钓鱼识别DDOS压力测试邮件反制分析应用日志

目录 前言&#xff1a; 1、#内网应急-日志分析-爆破&横向&数据库 2、#红队APT-钓鱼邮件识别-内容&发信人&附件 3、#拒绝服务攻击-DDOS&CC-代理&防火墙防御 用途&#xff1a;个人学习笔记&#xff0c;欢迎指正&#xff01; 前言&#xff1a; 了解和…

DVI接口如何连接HDMI接口显示器?DVI转HDMI转换器DHA

DVI转HDMI转换器DHA简介 DVI转HDMI转换器DHA能够将DVI信号和R/L音频信号输入转换成HDMI信号输出,独特的功能使其顺畅地整合到家庭影院中&#xff0c;并且播放出高品质的图像。主要用于数据监控中心、大型会议展示中心、学校及各个公司 DVI转HDMI转换器DHA特点 01.支持分辨率4K…

分治 (地毯填补问题)

地毯填补问题 题目描述 相传在一个古老的阿拉伯国家里&#xff0c;有一座宫殿。宫殿里有个四四方方的格子迷宫&#xff0c;国王选择驸马的方法非常特殊&#xff0c;也非常简单&#xff1a;公主就站在其中一个方格子上&#xff0c;只要谁能用地毯将除公主站立的地方外的所有地…

开发数据产品+AI产品通关上岸课程

该课程全面解析数据产品和人工智能产品的开发与设计。学员将学习产品规划、数据分析以及AI技术应用&#xff0c;通过案例实践掌握产品开发流程&#xff0c;致力于帮助他们成功进入数据和人工智能产品领域。 课程大小&#xff1a;9.8G 课程下载&#xff1a;https://download.cs…

【HarmonyOS】鸿蒙开发之HTTP网络请求——第5章

HTTP网络请求封装 network/request.ets import { configInterface } from ./type import http from ohos.net.http import { getToken } from ../utils/storage//网络请求封装 export const request (config:configInterface)>{let httpRequest:http.HttpRequest http.c…

汽车销量可视化分析

目录 一.分析的背景、目的、意义 1、背景 2、目的 3、意义 二.数据来源 三.图表分析 1、汽车品牌销量柱状图 2、中国汽车销量柱状图 3、汽车销量前10排行柱状图 4、汽车厂商销量折线图 ​编辑5、汽车销量词云图 6、汽车车型销量 7、汽车价格分布雷达图 8、汽车分…