使用 Pipeline 提高 Redis 批量操作性能
使用 Pipeline 提高 Redis 批量操作性能
在 Redis 中,Pipeline(管道) 是一种用于提高批量操作性能的技术。它允许客户端一次性发送多个命令到 Redis 服务器,而不需要等待每个命令的单独响应,从而减少了**网络往返(RTT, Round Trip Time)**的影响,显著提升性能。
为什么使用 Pipeline?
通常,在 Redis 客户端执行命令时,每条命令都需要:
- 客户端发送请求给 Redis 服务器。
- 服务器处理请求并返回结果。
- 客户端接收结果后,再发送下一条命令。
当需要执行大量命令时,传统的逐条请求方式会产生大量的 网络往返延迟(RTT)。例如,在 100ms 的网络延迟下,每秒最多只能执行 10 条命令(1000ms / 100ms)。
使用 Pipeline,可以:
- 批量发送命令,减少网络往返次数。
- 更快地执行大量命令,特别适用于写入操作(如
SET)。 - 降低 CPU 和 I/O 开销,提高吞吐量。
如何使用 Pipeline
Pipeline 的使用方法因编程语言的不同而有所区别,下面以 Python(redis-py)和 Node.js(ioredis)为例进行详细讲解。
1. 在 Python 中使用 Pipeline
Python 使用 redis-py 客户端,提供 pipeline() 方法来执行批量命令。
示例 1:基本 Pipeline 操作
import redis# 连接 Redis
r = redis.Redis(host='localhost', port=6379, db=0)# 创建 Pipeline
pipe = r.pipeline()# 批量执行 SET 命令
pipe.set("key1", "value1")
pipe.set("key2", "value2")
pipe.set("key3", "value3")# 执行 Pipeline(发送到 Redis 服务器并执行)
pipe.execute()# 验证是否成功
print(r.get("key1")) # b'value1'
print(r.get("key2")) # b'value2'
print(r.get("key3")) # b'value3'
解释:
pipeline()创建一个 Pipeline 对象。pipe.set()添加多个SET命令到 Pipeline,但并未立即执行。pipe.execute()统一发送到 Redis 服务器执行,提高性能。
示例 2:带返回值的 Pipeline
Pipeline 支持批量获取返回值:
pipe = r.pipeline()pipe.set("key4", "value4")
pipe.get("key4")
pipe.incr("counter") # 递增操作results = pipe.execute()print(results) # [True, b'value4', 1]
解释:
pipe.get("key4")会返回b'value4'。pipe.incr("counter")返回递增后的值。execute()返回所有命令的执行结果。
示例 3:批量写入
在处理大量数据时,Pipeline 可以显著提升效率:
pipe = r.pipeline()
for i in range(10000):pipe.set(f"key:{i}", f"value:{i}")
pipe.execute()
普通方式 vs Pipeline:
- 普通方式:每次
SET需要一次请求,10000 次请求开销很大。 - Pipeline:只需要很少的网络交互,提高吞吐量。
2. 在 Node.js(ioredis)中使用 Pipeline
Node.js 中 ioredis 提供了 pipeline() 方法,可以高效地批量执行 Redis 命令。
示例 1:基本 Pipeline 操作
const Redis = require("ioredis");
const redis = new Redis();const pipeline = redis.pipeline();pipeline.set("key1", "value1");
pipeline.set("key2", "value2");
pipeline.get("key1");pipeline.exec((err, results) => {console.log(results); // [[null, 'OK'], [null, 'OK'], [null, 'value1']]
});
解释:
redis.pipeline()创建 Pipeline。pipeline.set()和pipeline.get()只是加入队列,并未立即执行。exec()发送所有命令,返回结果。
示例 2:批量写入
const pipeline = redis.pipeline();
for (let i = 0; i < 10000; i++) {pipeline.set(`key:${i}`, `value:${i}`);
}
pipeline.exec().then(results => {console.log("Pipeline 批量写入完成");
});
普通方式 vs Pipeline:
- 普通方式:每次
set都会等待 Redis 响应,网络延迟大。 - Pipeline:减少网络请求次数,提高吞吐量。
3. Pipeline vs. MULTI/EXEC(事务)
Pipeline 不是事务,它只减少了网络往返次数,而 MULTI/EXEC 是 Redis 事务机制。
pipe = r.pipeline()
pipe.multi() # 开始事务
pipe.set("keyA", "valueA")
pipe.set("keyB", "valueB")
pipe.execute() # 事务内命令原子执行
区别:
| 特性 | Pipeline | MULTI/EXEC |
|---|---|---|
| 作用 | 批量减少网络往返 | 保证事务原子性 |
| 是否保证原子性 | 否 | 是 |
| 适用场景 | 高吞吐批量操作 | 严格事务要求 |
4. Pipeline vs. Lua 脚本
如果 多个操作之间有逻辑依赖,Pipeline 可能不适用。可以使用 Lua 脚本 代替:
script = '''
redis.call('SET', KEYS[1], ARGV[1])
redis.call('SET', KEYS[2], ARGV[2])
return redis.call('GET', KEYS[1])
'''
result = r.eval(script, 2, "keyX", "keyY", "valueX", "valueY")
print(result) # "valueX"
Lua 脚本 vs Pipeline
- Lua 脚本:原子执行,适用于有逻辑依赖的场景。
- Pipeline:适用于独立的批量操作。
5. Pipeline 性能测试
import redis
import timer = redis.Redis(host='localhost', port=6379, db=0)# 普通方式
start = time.time()
for i in range(10000):r.set(f"key:{i}", f"value:{i}")
end = time.time()
print(f"普通方式耗时: {end - start:.3f} 秒")# Pipeline 方式
start = time.time()
pipe = r.pipeline()
for i in range(10000):pipe.set(f"key:{i}", f"value:{i}")
pipe.execute()
end = time.time()
print(f"Pipeline 耗时: {end - start:.3f} 秒")
测试结果(示例):
普通方式耗时: 1.543 秒
Pipeline 耗时: 0.120 秒
Pipeline 速度提升了 10 倍以上!
总结
| 方法 | 适用场景 | 优势 | 劣势 |
|---|---|---|---|
| Pipeline | 高吞吐批量操作 | 减少网络往返,提高性能 | 不能保证原子性 |
| 事务(MULTI/EXEC) | 需要原子操作的场景 | 保证事务原子性 | 仍有网络延迟 |
| Lua 脚本 | 有逻辑依赖的复杂操作 | 原子执行,性能高 | 代码复杂度较高 |
最佳实践
- 批量写入时,使用 Pipeline。
- 需要原子操作时,使用 事务(MULTI/EXEC)。
- 复杂逻辑依赖时,使用 Lua 脚本。
这样,你可以高效地使用 Redis Pipeline 来优化你的应用! 🚀
相关文章:
使用 Pipeline 提高 Redis 批量操作性能
使用 Pipeline 提高 Redis 批量操作性能 在 Redis 中,Pipeline(管道) 是一种用于提高批量操作性能的技术。它允许客户端一次性发送多个命令到 Redis 服务器,而不需要等待每个命令的单独响应,从而减少了**网络往返&…...
「 机器人 」利用冲程对称性调节实现仿生飞行器姿态与方向控制
前言 在仿生扑翼飞行器中,通过改变冲程对称性这一技术手段,可以在上冲与下冲两个阶段引入不对称性,进而产生额外的力或力矩,用于实现俯仰或其他姿态方向的控制。以下从原理、在仿生飞行器中的应用和典型实验示例等方面进行梳理与阐述。 1. 冲程对称性原理 1.1 概念:上冲与…...
第十五届蓝桥杯大赛软件赛省赛C/C++ 大学 B 组
第十五届的题目在规定时间内做出了前5道,还有2道找时间再磨一磨。现在把做的一些思路总结如下: 题1:握手问题 问题描述 小蓝组织了一场算法交流会议,总共有 50人参加了本次会议。在会议上,大家进行了握手交流。按照惯例…...
本地大模型编程实战(02)语义检索(1)
文章目录 准备加载文档分割文档嵌入矢量存储查询矢量库检索返回评分先嵌入查询文本再检索 检索器总结代码 我们在百度、必应、谷歌等搜索引擎中使用的检索都是基于字符串的:用户输入字符串后,搜索引擎先对搜索内容进行分词,然后在已经进行了倒…...
自定义命令执行器:C++中命令封装的深度探索(C/C++实现)
在现代软件开发中,执行系统命令是一项常见的需求,无论是自动化脚本、系统管理工具,还是需要调用外部程序的复杂应用程序,都离不开对系统命令的调用。然而,直接使用系统调用(如 execve)虽然简单&…...
C语言程序设计十大排序—选择排序
文章目录 1.概念✅2.选择排序🎈3.代码实现✅3.1 直接写✨3.2 函数✨ 4.总结✅5.十大排序 1.概念✅ 排序是数据处理的基本操作之一,每次算法竞赛都很多题目用到排序。排序算法是计算机科学中基础且常用的算法,排序后的数据更易于处理和查找。在…...
C语言初阶牛客网刷题——HJ73 计算日期到天数转换【难度:简单】
1. 题目描述——HJ73 计算日期到天数转换 牛客网OJ题链接 描述 每一年中都有 12 个月份。其中,1,3,5,7,8,10,12 月每个月有 31 天; 4,6,9,11 月每个月有 30 天;而对于 2 月,闰年时有29 天,平年时有 28 天。 现在&am…...
MATLAB中alphanumericsPattern函数用法
目录 语法 说明 示例 从文本中提取字母和数字 匹配所设置数目的字母和数字 匹配不同大小的字母和数字集合 alphanumericsPattern函数的功能是匹配字母和数字字符。 语法 pat alphanumericsPattern pat alphanumericsPattern(N) pat alphanumericsPattern(minCharact…...
人工智能在教育领域的创新应用与前景展望
人工智能在教育领域的创新应用与前景展望 摘要:本文围绕人工智能在教育领域的应用展开深入探讨,分析了人工智能为教育带来的创新变革,如个性化学习支持、智能教学辅助、教育资源优化等方面的显著成效,同时探讨了其在应用过程中面…...
约束布局 ConstraintLayout
鸿蒙操作系统(HarmonyOS)是华为公司自主研发的面向全场景的分布式操作系统,旨在为用户提供流畅、安全、可靠的智能生活体验。随着鸿蒙操作系统的不断发展和完善,越来越多的开发者开始关注并投入到鸿蒙应用开发中来。对于想要深入理…...
校园商铺管理系统设计与实现(代码+数据库+LW)
摘 要 信息数据从传统到当代,是一直在变革当中,突如其来的互联网让传统的信息管理看到了革命性的曙光,因为传统信息管理从时效性,还是安全性,还是可操作性等各个方面来讲,遇到了互联网时代才发现能补上自…...
react native在windows环境搭建并使用脚手架新建工程
截止到2024-1-11,使用的主要软件的版本如下: 软件实体版本react-native0.77.0react18.3.1react-native-community/cli15.0.1Android Studio2022.3.1 Patch3Android SDKAndroid SDK Platform 34 35Android SDKAndroid SDK Tools 34 35Android SDKIntel x…...
vulnhub DC-1靶机 walkthrough
描述 DC-1 是专门为获得渗透测试经验而建造的易受攻击实验室。 它旨在成为初学者的挑战,但其难易程度取决于您的技能和知识以及学习能力。 要成功完成此挑战,您需要具备 Linux 技能、熟悉 Linux 命令行以及使用基本渗透测试工具的经验,例如 …...
计算机网络 (58)无线局域网WLAN
前言 无线局域网WLAN(Wireless Local Area Network)是一种利用无线通信技术将计算机设备互联起来,构成可以互相通信和实现资源共享的网络体系。 一、定义与特点 定义: WLAN通过无线信道代替有线传输介质连接两个或多个设备形成一个…...
基于物联网的风机故障检测装置的设计与实现
1 系统总体设计方案 通过对风机故障检测装置的设计与实现的需求、可行性进行分析,本设计风机故障检测装置的设计与实现的系统总体架构设计如图2-1所示,系统风机故障检测装置采用STM32F103单片机作为控制器,并通过DS18B20温度传感器、ACS712电…...
【AI日记】25.01.25
【AI论文解读】【AI知识点】【AI小项目】【AI战略思考】【AI日记】【读书与思考】 AI kaggle 比赛:Forecasting Sticker Sales 读书 书名:法治的细节 律己 AI:8 小时,良作息:00:30-8:30, 良短视频&…...
Sourcetree:一款高效便捷的Git版本控制客户端
Sourcetree:一款高效便捷的Git版本控制客户端 引言 在软件开发领域,版本控制是确保代码质量和项目进度的关键工具。Git作为当前最流行的版本控制系统,已经广泛应用于各类项目中。而Sourcetree作为Git的图形化界面客户端,凭借其易…...
图像处理算法研究的程序框架
目录 1 程序框架简介 2 C#图像读取、显示、保存模块 3 C动态库图像算法模块 4 C#调用C动态库 5 演示Demo 5.1 开发环境 5.2 功能介绍 5.3 下载地址 参考 1 程序框架简介 一个图像处理算法研究的常用程序逻辑框架,如下图所示 在该框架中,将图像处…...
GitLab配置免密登录和常用命令
SSH 免密登录 Windows免密登录 删除现有Key 访问目录:C:\Users\Administrator\ .ssh,删除公钥:id_rsa.pub ,私钥:id_rsa 2.生成.ssh 秘钥 运行命令生成.ssh 秘钥目录( ssh-keygen -t rsa -C xxxxxx126.…...
Chameleon(变色龙) 跨平台编译C文件,并一次性生成多个平台的可执行文件
地址:https://github.com/MartinxMax/Chameleon Chameleon 跨平台编译C文件,并一次性生成多个平台的可执行文件。可以通过编译Chameleon自带的.C文件反向Shell生成不同平台攻击载荷。 登录 & 代理设置 按照以下步骤设置 Docker 的代理: 创建配置目…...
Spring Boot 实现流式响应(兼容 2.7.x)
在实际开发中,我们可能会遇到一些流式数据处理的场景,比如接收来自上游接口的 Server-Sent Events(SSE) 或 流式 JSON 内容,并将其原样中转给前端页面或客户端。这种情况下,传统的 RestTemplate 缓存机制会…...
相机Camera日志实例分析之二:相机Camx【专业模式开启直方图拍照】单帧流程日志详解
【关注我,后续持续新增专题博文,谢谢!!!】 上一篇我们讲了: 这一篇我们开始讲: 目录 一、场景操作步骤 二、日志基础关键字分级如下 三、场景日志如下: 一、场景操作步骤 操作步…...
Leetcode 3577. Count the Number of Computer Unlocking Permutations
Leetcode 3577. Count the Number of Computer Unlocking Permutations 1. 解题思路2. 代码实现 题目链接:3577. Count the Number of Computer Unlocking Permutations 1. 解题思路 这一题其实就是一个脑筋急转弯,要想要能够将所有的电脑解锁&#x…...
【ROS】Nav2源码之nav2_behavior_tree-行为树节点列表
1、行为树节点分类 在 Nav2(Navigation2)的行为树框架中,行为树节点插件按照功能分为 Action(动作节点)、Condition(条件节点)、Control(控制节点) 和 Decorator(装饰节点) 四类。 1.1 动作节点 Action 执行具体的机器人操作或任务,直接与硬件、传感器或外部系统…...
python如何将word的doc另存为docx
将 DOCX 文件另存为 DOCX 格式(Python 实现) 在 Python 中,你可以使用 python-docx 库来操作 Word 文档。不过需要注意的是,.doc 是旧的 Word 格式,而 .docx 是新的基于 XML 的格式。python-docx 只能处理 .docx 格式…...
在鸿蒙HarmonyOS 5中使用DevEco Studio实现录音机应用
1. 项目配置与权限设置 1.1 配置module.json5 {"module": {"requestPermissions": [{"name": "ohos.permission.MICROPHONE","reason": "录音需要麦克风权限"},{"name": "ohos.permission.WRITE…...
【JavaWeb】Docker项目部署
引言 之前学习了Linux操作系统的常见命令,在Linux上安装软件,以及如何在Linux上部署一个单体项目,大多数同学都会有相同的感受,那就是麻烦。 核心体现在三点: 命令太多了,记不住 软件安装包名字复杂&…...
Go 语言并发编程基础:无缓冲与有缓冲通道
在上一章节中,我们了解了 Channel 的基本用法。本章将重点分析 Go 中通道的两种类型 —— 无缓冲通道与有缓冲通道,它们在并发编程中各具特点和应用场景。 一、通道的基本分类 类型定义形式特点无缓冲通道make(chan T)发送和接收都必须准备好࿰…...
C++.OpenGL (20/64)混合(Blending)
混合(Blending) 透明效果核心原理 #mermaid-svg-SWG0UzVfJms7Sm3e {font-family:"trebuchet ms",verdana,arial,sans-serif;font-size:16px;fill:#333;}#mermaid-svg-SWG0UzVfJms7Sm3e .error-icon{fill:#552222;}#mermaid-svg-SWG0UzVfJms7Sm3e .error-text{fill…...
华为OD机考-机房布局
import java.util.*;public class DemoTest5 {public static void main(String[] args) {Scanner in new Scanner(System.in);// 注意 hasNext 和 hasNextLine 的区别while (in.hasNextLine()) { // 注意 while 处理多个 caseSystem.out.println(solve(in.nextLine()));}}priv…...
