首发于 樊浩柏科学院

Redis 的 管道 (pipelining)是用来打包多条无关命令批量执行,以减少多个命令分别执行带来的网络交互时间。在一些批量操作数据的场景,使用管道可以显著提升 Redis 的读写性能。

原理演示

Redis 的管道实质就是命令打包批量执行,多次网络交互减少到单次。使用管道和不使用管道时的交互过程如下:

我们使用 nc 命令来直观感受下 Redis 管道的使用过程:

# 安装nc命令
$ yum install nc
# nc打包多个命令
$ (printf "PING\r\nPING\r\nPING\r\n") | nc localhost 6379
# 响应
+PONG
+PONG
+PONG
``` 因此,只要通过管道进行命令打包后,Redis 就可以批量返回命令的执行结果了。

管道的应用

首先,构造示例需要的 Hash 用户数据:
$keyPrex = 'user:hash:u:';
for ($i=1; $i<=10000; $i++) {
$redis->hMset($keyPrex.$i, [
'name' => name(), //name()函数生成随机姓名
'age' => rand(21, 30),
'sex' => rand(0, 1),
'is_new' => rand(0, 1)
]);
}
``` 然后,查看导入 Redis 中的数据:
127.0.0.1:6379> keys user:hash:u:*
9997) "user:hash:u:3013"
9998) "user:hash:u:8971"
9999) "user:hash:u:4761"
10000) "user:hash:u:1828" 127.0.0.1:6379> HGETALL user#️⃣u:1828
  1. "name"
  2. "ggrg"
  3. "age"
  4. "23"
  5. "sex"
  6. "0"
  7. "is_new"
  8. "1"

<h3>需求</h3>
<p>在某个社交活动中,通过一系列筛选逻辑后取得种子用户 uid,然后用这些 uid 去 Hash 获取用户的信息。这种情况下你会怎么来处理呢?</p>
<h3>不使用管道</h3>
<p>一般情况下,在数据量较小时,我们会直接使用 HGETALL 命令遍历地获取用户数据。</p>
<pre><code class="PHP">$start = nowTime();
foreach (range(1, 1000) as $id) {
$user[] = $redis-&gt;hgetAll($keyPrex.$id);
}
echo '时间:', nowTime() - $start, 'ms', PHP_EOL; 时间:39ms

执行所用时间:39ms

使用管道

因为通过 uid 批量获取用户数据,各个命令并没有依赖关系,所以可以使用 Redis 的管道来优化查询。

$start = nowTime();
$redis->multi(Redis::PIPELINE);
foreach (range(1, 1000) as $id) {
//返回资源id相同的socket资源,并未执行命令
$redis->hgetAll($keyPrex.$id);
}
$user = $redis->exec();
echo '时间:', nowTime() - $start, 'ms', PHP_EOL; 时间:6ms

<p>使用管道后,执行时间显著地减少为:<a href="#">6ms</a>。使用 tcpdump 抓取打包后的命令如下:</p>
<pre><code class="Tcp">10:45:03.029049 IP localhost.58176 &gt; localhost.6379: Flags [P.], seq 2255478840:2255479211, ack 3144685411, win 342, options [nop,nop,TS val 17640474 ecr 17640474], length 371
E..../@.@.o..........@...o.8.p.c...V.......
,.*2
$7
HGETALL
$13
user:hash:u:1
*2
$7
HGETALL
$13
user:hash:u:2
*2
$7
... ...

适用场景

在批量操作(查询和写入)数据时,我们应尽量避免多次跟 Redis 的网络交互。这时,可以使用管道实现,也可以 Redis 内嵌 Lua 脚本实现。需要注意的是

  • 管道只适用于无因果关联的多命令操作,否则就需要借助 Lua 脚本实现批量操作;
  • 在实际应用中,Redis 往往不可能是单机部署,如果想要在集群中使用管道,可以部署为一主多从架构,此时所有节点的数据都一致,随机选取节点使用管道即可;

总结

在批量获取数据时,尽管使用 Redis 的管道性能会显著提升,但是使用管道时 Redis 会缓存之前命令的结果,最后一并输出给终端,因此所打包的命令不宜太多,否则内存使用会很严重。

            原文地址:

最新文章

  1. 执行后台任务的利器&mdash;&mdash;Hangfire
  2. mySql 注入攻击
  3. python chinese code
  4. uoot启动过程
  5. Redis+Django(Session,Cookie)的用户系统
  6. C# 时间现实问题(12小时制与24小时制)
  7. springMVC+hibernate构建项目
  8. 理解SVG坐标系和变换:视窗,viewBox和preserveAspectRatio
  9. Maven多层嵌套
  10. 集合ArrayList双色球练一练(自己的方法,太麻烦)
  11. Quartz与Spring集成 Job如何自动注入Spring容器托管的对象
  12. 【已解决】通过Package或者Package+Activity启动应用
  13. 其它综合-运维老鸟分享linux运维发展路线规划
  14. gdb 命令汇总
  15. Jenkins环境搭建(2)-搭建jmeter+ant+jenkins自动化测试环境
  16. rabbitmq web管理
  17. python 对时间操作
  18. [py]requests+json模块处理api数据,flask前台展示
  19. docker搭建nginx
  20. js 获取地址栏的值乱码问题

热门文章

  1. 解决使用mybatis模糊查询为空的问题
  2. C语言内存布局简记待补充
  3. Windows API 第六篇 GetLocalTime
  4. 廖雪峰Java10加密与安全-3摘要算法-5Hmac
  5. MaxCompute用户初体验
  6. HZOI20190814 B 不等式
  7. oauth2使用心得-----基本概念以及认证服务器搭建
  8. 转载 ASP.NET SignalR 与LayIM配合,轻松实现网站客服聊天室(一) 整理基础数据
  9. 【DM642学习笔记六】TI参考文档--DM642 Video Port Mini Driver
  10. idea短信验证