Redis中的Scan命令的使用
Redis中有一个经典的问题,在巨大的数据量的情况下,做类似于查找符合某种规则的Key的信息,这里就有两种方式,
一是keys命令,简单粗暴,由于Redis单线程这一特性,keys命令是以阻塞的方式执行的,keys是以遍历的方式实现的复杂度是 O(n),Redis库中的key越多,查找实现代价越大,产生的阻塞时间越长。
二是scan命令,以非阻塞的方式实现key值的查找,绝大多数情况下是可以替代keys命令的,可选性更强
以下写入100000条key***:value***格式的测试数据(ps:用pipline的话,1w一笔,每一笔在秒级完成)
# -*- coding: utf-8 -*-
# !/usr/bin/env python3
import redis
import sys
import datetimedef create_testdata():r = redis.StrictRedis(host='***.***.***.***', port=***, db=0, password='***')counter = 0with r.pipeline(transaction=False) as p:for i in range(0, 100000):p.set('key' + str(i), "value" + str(i))counter = counter + 1if (counter == 10000):p.execute()counter = 0print("set by pipline loop")if __name__ == "__main__":create_testdata()
比如这里查询key111开头的key有哪些?
若使用keys命令,则执行keys key1111*,一次性全部查出来。
同样,如果使用scan命令,则用 scan 0 match key1111* count 20
scan的语法为:SCAN cursor [MATCH pattern] [COUNT count] The default COUNT value is 10.
SCAN命令是一个基于游标的迭代器。这意味着命令每次被调用都需要使用上一次这个调用返回的游标作为该次调用的游标参数,以此来延续之前的迭代过程。
这里使用scan 0 match key1111* count 20命令来完成这个查询,稍显意外的是,使用一开始都没有查询到结果,这个要从scan命令的原理来看。
scan在遍历key的时候,0就代表第一次,key1111*代表按照key1111开头的模式匹配,count 20中的20并不是代表输出符合条件的key,而是限定服务器单次遍历的字典槽位数量(约等于)。
那么,什么又叫做槽的数据?这个槽是不是Redis集群中的slot?答案是否定的。其实上图已经给出了答案了。
如果上面说的“字典槽”的数量是集群中的slot,又知道集群中的slot数量是16384,那么遍历16384个槽之后,必然能遍历出来所有的key信息,
上面清楚地看到,当遍历的字典槽的数量20000的时候,游标依旧没有走完遍历结果,因此这个字典槽并不等于集群中的slot的概念。
经过测试,在scan的时候,究竟遍历多大的COUNT值能完全match到符合条件的key,跟具体对象的key的个数有关,
如果以超过key个数的count来scan,必定会一次性就查找到所有符合条件的key,比如在key个数为10W个的情况下,一次遍历20w个字典槽,肯定能完全遍历出来结果。
scan 指令是一系列指令,除了可以遍历所有的 key 之外,还可以对指定的容器集合进行遍历。
zscan 遍历 zset 集合元素,
hscan 遍历 hash 字典的元素、
sscan 遍历 set 集合的元素。
SSCAN 命令、 HSCAN 命令和 ZSCAN 命令的第一个参数总是一个数据库键(某个指定的key)。
另外,使用redis desktop manager的时候,当刷新某个库的时候,控制台自动不断刷新scan命令,也就知道它在干嘛了
Redis中的Scan命令的使用相关推荐
- Redis中的执行命令的过程
Redis中的执行命令的过程 在redis.c的initServerConfig()方法中,通过调用dictCreate方法初始化server端的命令表.这个命令表是一个hashtable,可以通过k ...
- redis中KEYS替代命令
在 Redis 中,还有哪些其他命令可以代替 KEYS 命令,实现同样的功能呢?这些命令的复杂度会导致 Redis 变慢吗? 如果想要获取整个实例的所有key,建议使用SCAN命令代替.客户端通过执行 ...
- php redis命令大全,redis中key相关命令详解
一.概述: 本文将主要讲述与Key相关的Redis命令.学习这些命令对于学习Redis是非常重要的基础,也是能够充分挖掘Redis潜力的利器.(推荐:redis视频教程) 二.相关命令列表: 命令原型 ...
- redis中的incr命令和incrby命令
Redis Incr 命令将 key 中储存的数字值增一,如果 key 不存在,那么 key 的值会先被初始化为 0 ,然后再执行 INCR 操作. Redis Incrby 命令将 key 中储存的 ...
- redis中的bitfield命令
redis3.2后新增了一个bitfield命令,可以一次对多个位进行操作.这个指令有三个子指令,get,set,incrby,都可以对指定位片段进行读写,但最多只能处理64个连续的位,如超过64位, ...
- Redis中的ttl命令用法解析
一.语法格式: ttl key 二.作用: 查看键还有多少秒过期. 三.返回值代表的意义:-1表示永不过期,-2表示已经过期.
- 【Redis】--- 不同数据结构命令
[Redis]--- 不同数据结构命令 前言 数据结构 string (字符串) 使用场景 命令 原子计数 list (列表) 使用场景 命令 hash (字典) 使用场景 命令 set (集合) 使 ...
- redis的scan命令的源码分析,实现原理
简言 1. 线上环境keys命令不可用,会导致redis卡死.scan命令因为可以分批遍历,比较实用 2. scan命令包括多个 遍历整个数据库的scan命令,处理函数 scanCommand(),最 ...
- 【Redis】详细基础命令 - 学习笔记
Redis 环境搭建及运行 安装(Ubuntu举例,其他系统类似) apt-get update && apt-get install redis-server -y 启动 redis ...
- 面试必问:如何访问 Redis 中的海量数据?
前言 有时候我们需要知道线上的redis的使用情况,尤其需要知道一些前缀的key值,那我们怎么去查看呢? 事故产生 因为我们的用户token缓存是采用了[user_token:userid]格式的ke ...
最新文章
- Zookeeper集群部署和使用
- 手指贴个“创可贴”,你睡觉都能发电
- Copy GAC DLL
- 华为怎么安装服务器系统版本,服务器怎么安装操作系统版本
- 循环内部异步函数处理相关问题解析
- 【LeetCode笔记】剑指 Offer 44. 数字序列中某一位的数字(Java、偏数学)
- Apache mod_rewrite
- 利用Runtime为Category添加属性
- 「leetcode」78. 子集【回溯算法】详解!
- vue电商后台管理项目总结
- OpenJudge NOI 2.1 1813:熄灯问题
- JS设置浏览器缩放比例
- 服务器协议和交换机怎么转换,服务器与交换机连接怎样配置
- excel多表合并为一个表
- FFmpeg windows 录屏(录像)录音 实测
- Redis是什么?有什么用?
- 计算机显示器图片怎么铺满全屏,win10电脑显示器屏幕不能铺满怎么办_win10电脑显示不能铺满屏幕处理方法-win7之家...
- 摩根大通区块链支付网络计划于1月日本启动
- 高德地图小程序步行路线显示_微信小程序 高德地图路线规划实现过程详解
- Generalized face anti-spoofing by detecting pulse from face videos