如何在Redis里按模式删除数据?

运维 系统运维 Redis
一台Redis服务器在很短的时间里消耗了几十个G的内存,最终因为SWAP而宕机。因为这台服务器的社会背景比较复杂,所以一时无法判断犯罪嫌疑人到底是谁。

一台Redis服务器在很短的时间里消耗了几十个G的内存,最终因为SWAP而宕机。因为这台服务器的社会背景比较复杂,所以一时无法判断犯罪嫌疑人到底是谁。

最开始的直觉是认为肯定有人保存了大体积的数据,于是问题就变成了找出哪些键占用的空间比较大,DBA同事用了redis-rdb-tools等工具来分析数据文件。可惜的是虽然找到了一些大体积的键,但最终都排除了嫌疑,问题似乎陷入了僵局。

在被直觉带入死胡同之后,我们开始调整调查的角度:即便一个键本身占用的空间并不大,但是如果相同模式的键数量很多的话,那么合计起来一样会占用大量空间,于是问题就变成了找出哪些相同模式的键占用的空间比较大。这次我不想用什么工具,而是打算在测试服务器上一边删除可疑键一边查看内存变化情况:

shell> /path/to/redis-cli keys foo:* | xargs /path/to/redis-cli del

悲催的是一运行这个命令服务器就挂了!因为数据太多了,所以KEYS受不了。此时应该使用SCAN,它有游标的概念,每次迭代只涉及很少的数据。

直接在命令行使用SCAN有些麻烦,于是我用了PHP

<?php

$redis = new Redis();
$redis->setOption(Redis::OPT_SCAN, Redis::SCAN_RETRY);

$match = 'foo:*';
$count = 10000;

while ($keys = $redis->scan($it, $match, $count)) {
    $redis->del($keys);
}

?>

在删除的同时注意监控内存变化情况,就能确认问题了:

shell> watch -d -n 1 '/path/to/redis-cli info | grep memory'

至于可疑键的获取,我是瞎蒙的,简单通过MONITOR或者SCAN获取采样数据即可,另外从此案例看,监控键总数的变化幅度是很重要的,从INFO里能拿到它。

责任编辑:黄丹 来源: 火丁笔记
相关推荐

2023-09-05 08:40:57

删除数据库Oracle

2022-01-02 08:38:22

Redis数据单线程

2010-09-30 09:42:22

DB2删除数据

2021-08-09 11:31:54

Linux重复相片删除

2011-05-18 15:08:03

mysql删除修改数据

2011-02-21 10:35:00

查询删除数据

2009-09-04 17:56:22

C#删除数据

2019-12-20 14:56:50

批量删除数据数据删除

2009-11-25 16:29:08

PHP删除数组元素

2011-02-21 14:10:33

QmailSMTP

2021-03-22 08:51:59

RedisKey数据库

2010-09-02 10:15:46

SQL删除

2021-04-30 13:19:20

Linux删除分区

2011-01-21 10:50:53

Sendmail

2010-10-22 16:40:27

SQL TRUNCAT

2009-09-14 10:45:33

LINQ删除数据

2014-12-22 09:48:58

DBCAOracle 11

2017-04-01 18:30:47

MySQL误删除数据库

2013-12-11 10:21:31

2018-04-08 08:32:59

点赞
收藏

51CTO技术栈公众号