如何优雅地删除Redis set集合中前缀相同的key?
      Redis中有删除单条数据的命令DEL,却没有批量删除特定前缀key的指令,但我们经常遇到需要根据前缀来删除的业务场景,那么究竟该怎么做呢?可能你一通搜索后会得到下边的答案:
redis-cli --raw keys "prefix-*" | xargs redis-cli del

直接在linux下通过redis的keys命令匹配到所有的key,然后调用系统命令xargs来删除,看似十全十美,实则风险巨大。这就是一颗随时爆炸的炸弹!我们都知道Redis是单线程服务模式,使用命令 keys * 查询key的时候会阻塞正常的业务请求,甚至造成redis宕机,所以肯定不行。因此,我们在生产环境中应当避免使用上边的方法,那使用什么优雅的方法来解决呢?SCAN!

SCAN介绍

       Redis从2.8版本开始支持SCAN命令,它是一个基于游标的迭代器,每次被调用之后都会返回一个新的游标,用户在下次迭代时需要使用这个新游标作为SCAN命令的游标参数,以此来延续之前的迭代过程,直到服务器返回值为0的游标时,一次完整的遍历过程就结束了。
       SCAN命令的基本语法如下:
 
scan cursor [MATCH pattern] [COUNT count]

MATCH: 匹配规则,例如遍历以ops-coffee-开头的所有key可以写成ops-coffee-*,中间包含-coffee-的可以写成*-coffee-*cursor: 游标,

COUNT:  COUNT选项的作用就是让用户告知迭代命令,在每次迭代中应该从数据集里返回多少元素,COUNT只是对增量式迭代命令的一种提示,并不代表真正返回的数量,例如你COUNT设置为2有可能会返回3个元素,但返回的元素数据会与COUNT设置的正相关,COUNT的默认值是10。
     下面在jedis中操作scan和keys,以删除相同前缀的key。
 
public void testSetDel(Jedis jedis) {
try {
log.info("---------------- Tests begin -----------------");
initRedisData(jedis);
String givenKey = "prefix_*";
delValuesByKeys(givenKey, jedis); log.info("开始使用 scan 删除数据 ------------ ");
initRedisData(jedis);
this.delSetValues(givenKey, jedis); Set<String> keys = jedis.keys(givenKey);
log.info("---------------- Tests end ----------------- 是否存在相同前缀的key result = " + !CollectionUtils.isEmpty(keys));
} catch (Exception e) {
log.error(" 删除指定前缀的key对应的键值对 " + e);
} finally {
if (jedis != null) {
jedis.close();
log.info("关闭jedis连接");
}
}
}
/**
* java redis 删除指定前缀的key对应的键值对
* @param givenKey
* @return
*/
public Boolean delSetValues(String givenKey, Jedis jedis) throws Exception {
log.info("开始模糊删除set中的数据,givenKey = " + givenKey);
List<String> keys = getByScan(givenKey, jedis);
log.info("即将删除的key是 " + keys);
String[] array = keys.toArray(new String[0]);
jedis.del(array);
return true;
}
/**
* Jedis 删除指定前缀的key对应的key,使用keys
* @param givenKey
* @return
*/
public Boolean delValuesByKeys(String givenKey, Jedis jedis) throws Exception {
log.info("开始模糊删除set中的数据,givenKey = " + givenKey);
Set<String> keys = jedis.keys(givenKey);
for (String key : keys) {
log.info("当前 key 是 :" + key);
jedis.del(key);
}
return true;
}
private void initRedisData(Jedis jedis) {
jedis.set("prefix_1333", "1");
jedis.set("prefix_2KKKKK", "2");
jedis.set("prefix_3哈哈哈哈哈哈", "777");
}

// 使用 scan
public List<String> getByScan(String key, Jedis jedis) {
List<String> list = new ArrayList<>();
ScanParams params = new ScanParams();
params.match(key);
params.count(100);
String cursor = "0";
while (true) {
ScanResult scanResult = jedis.scan(cursor, params);
List<String> eles = scanResult.getResult();
if (!CollectionUtils.isEmpty(eles)) {
list.addAll(eles);
}
cursor = scanResult.getStringCursor();
if ("0".equals(cursor)) {
break;
}
}
log.info(" getByScan 查到的数据集是 ============ " + list);
return list;
}

测试结果:

---------------- Tests begin -----------------
开始模糊删除set中的数据,givenKey = prefix_*
当前 key 是 :prefix_1333
当前 key 是 :prefix_3哈哈哈哈哈哈
当前 key 是 :prefix_2KKKKK
开始使用 scan 删除数据 ------------
开始模糊删除set中的数据,givenKey = prefix_*
getByScan 查到的数据集是 ============ [prefix_1333, prefix_2KKKKK, prefix_3哈哈哈哈哈哈]
即将删除的key是 [prefix_1333, prefix_2KKKKK, prefix_3哈哈哈哈哈哈]
---------------- Tests end ----------------- 是否存在相同前缀的 key result = false
   总结:虽然不提倡使用keys命令删除key,但是,本文示例依然给出了示例,目的在于了解使用原理。当然,比使用scan命令删除key效果更好的方案是直接调用Lua脚本,童鞋们自己琢磨吧!
 

Redis删除相同前缀的key的更多相关文章

  1. laravel redis 删除指定前缀的 key

    // 前缀 $prefix = 'abc'; // 需要在前面连接上应用的缓存前缀 $keys = app('redis')->keys(config('cache.prefix') . $pr ...

  2. Redis删除特定前缀key的优雅实现

    还在用keys命令模糊匹配删除数据吗?这就是一颗随时爆炸的炸弹! Redis中没有批量删除特定前缀key的指令,但我们往往需要根据前缀来删除,那么究竟该怎么做呢?可能你一通搜索后会得到下边的答案 re ...

  3. redis删除指定前缀的缓存

    redis作为缓存服务器为MySQL数据库提供较高的防御性,对于一些数据的查询可以直接从缓存中可以进行查询. 但是,某些情况下,我们需要清除缓存. 以下场景: 公司经常做活动,每个活动都存在大量的数据 ...

  4. php redis 获取指定前缀的所有key

    php redis 获取指定前缀的所有key 以laravel框架为例: $key = $this->redis->keys('db:shipping:shippingId:' . &qu ...

  5. redis 删除大key集合的方法

    redis大key,这里指的是大的集合数据类型,如(set/hash/list/sorted set),一个key包含很多元素.由于redis是单线程,在删除大key(千万级别的set集合)的时候,或 ...

  6. redis删除单个key和多个key,ssdb会落地导致重启redis无法清除缓存

    redis删除单个key和多个key,ssdb会落地导致重启redis无法清除缓存,需要针对单个key进行删除 删除单个:del key 删除多个:redis-cli -a pass(密码) keys ...

  7. redis key全量导出与导出指定前缀的key

    redis命令列表中有两种方法可以全量导出所有的key: (1)keys 由于redis是单线程的,使用keys会导致redis服务阻塞,不建议线上服务采用这种方式. (2)scan 命令,下面是使用 ...

  8. reids中删除某个前缀的所有key

    需求:reids中删除某个前缀的所有key 说明:代码中的0:2标识从key前缀中截取前2个字符,这里示例的时候比如“b_”前缀,使用时候根据实际情况截取对应的长度进行判断即可. 生成测试数据 #!/ ...

  9. redis如何清除所有的key

    redis比memcache好的地方之一,如果memcache,恐怕就得关掉重启了. 1 使用cli FLUSHDB 清除一个数据库,FLUSHALL清除整个redis数据. 2 使用shell re ...

随机推荐

  1. java中对List中的元素进行排序

    Collections对List集合中的数据进行排序 有时候需要对集合中的元素按照一定的规则进行排序,这就需要用到 Java中提供的对集合进行操作的工具类Collections,其中的sort方法 N ...

  2. Java Web-Ajax学习

    Java Web-Ajax学习 概念 Ajax(Asynchronous JavaScript And XML,异步的JavaScript和XML). 异步和同步:在客户端和服务器端相互通信的基础上来 ...

  3. GoAccess 视图化access.log 日志

    1.安装GoAccess 工具可以直接使用 apt-get install goaccess 2.使用goaccess命令将日志生成html文件 goaccess 日志路径 -o 输出HTML的路径 ...

  4. js 单引号和双引号相互替换的实现方法

    1.双引号替换成单引号 var domo = JSON.stringify(address).replace(/\"/g,"'"); var a = {a:1,b:2}; ...

  5. [ansible-playbook]4 持续集成环境之分布式部署利器 ansible playbook学习

    3 ansible-play讲的中太少了,今天稍微深入学习一点 预计阅读时间:15分钟 一: 安装部署 参考 http://getansible.com/begin/an_zhuang_ansile ...

  6. 如何修改配置文件:CentOS下SSH端口修改

    CentOS各发行版中SSH端口默认为22,如果正式做站或其它用途,为了提高安全性就需要修改掉默认的SSH端口号,防止被有心人穷举密码.部分VPS提供商,若您的VPS服务器SSH遭受多次的暴力破解,可 ...

  7. 如何使用Feign构造多参数的请求

    原文:http://www.itmuch.com/spring-cloud-sum/feign-multiple-params/ 本节来探讨如何使用Feign构造多参数的请求.笔者以GET及POST请 ...

  8. Luogu P1114 “非常男女”计划/Luogu P2697 宝石串

    Luogu P1114 "非常男女"计划/Luogu P2697 宝石串 (感觉我最近很爱做双倍经验的题啊) 使$d$等于第$i$个位置男生数(绿宝石数)减女生数(红宝石数)的差值 ...

  9. P2P system: Introduction

    P2P system : peer-to-peer system 一些流行的P2P system: Napster, Gnutella 我们为什么研究P2P system 大型的分布式系统有成千上万个 ...

  10. Java中list在循环中删除元素的坑

    JAVA中循环遍历list有三种方式for循环.增强for循环(也就是常说的foreach循环).iterator遍历. 1.for循环遍历list for(int i=0;i<list.siz ...