Redis SCAN命令操作实战(详细)

2023-09-20 13:28:00

目录

SCAN 介绍

SCAN 命令基本用法

MATCH 选项用法

COUNT 选项用法

TYPE 选项用法

补充

并发执行多个迭代

中途停止迭代

使用错误的游标进行增量式迭代

迭代终结的保证


SCAN 介绍

SCAN cursor [MATCH pattern] [COUNT count][TYPE type]:SCAN 命令及其相关的 SSCAN 命令、 HSCAN 命令和 ZSCAN 命令都用于增量地迭代(incrementally iterate)一集元素(a collection of elements)

  • MATCH选项:可以通过提供一个 glob 风格的模式参数, 让命令只返回和给定模式相匹配的元素
  • COUNT选项:增量式迭代命令不保证每次迭代所返回的元素数量, 但我们可以使用 COUNT 选项, 对命令的行为进行一定程度上的调整返回最大数量
  • TYPE选项:迭代列出指定的数据类型的元素

增量式迭代命令:

  • SCAN 命令用于迭代当前数据库中的数据库键。
  • SSCAN 命令用于迭代集合键中的元素。
  • HSCAN 命令用于迭代哈希键中的键值对。
  • ZSCAN 命令用于迭代有序集合中的元素(包括元素成员和元素分值)

以上命令每次执行均返回少量元素,而不像keys或smembers命令被用于处理大量数据是极有可能导致阻塞服务器。

SCAN 命令基本用法

SCAN 0:SCAN 命令是一个基于游标的迭代器: SCAN 命令每次被调用之后, 都会向用户返回一个新的游标, 用户在下次迭代时需要使用这个新游标作为 SCAN 命令的游标参数, 以此来延续之前的迭代过程

  • 当 SCAN 命令的游标参数被设置为 0 时, 服务器将开始一次新的迭代, 而当服务器向用户返回值为 0 的游标时, 表示迭代已结束
127.0.0.1:6379> mset a 1 b 1 c 1 d 1 e 1 f 1 g 1 h 1 i 1 j 1 k 1 l 1 m 1 n 1 o 1 p 1 q 1 r 1 s 1 t 1 u 1 v 1 w 1 x 1 y 1 z 1 # 初始化键值对
OK
127.0.0.1:6379> scan 0
1) "1"                    # 返回下次换代的新游标
2)  1) "b"
    2) "t"
    3) "m"
    4) "q"
    5) "e"
    6) "u"
    7) "g"
    8) "w"
    9) "h"
   10) "s"
127.0.0.1:6379> scan 1
1) "27"                   # 返回下次换代的新游标
2)  1) "z"
    2) "c"
    4) "l"
    5) "y"
    6) "d"
    7) "a"
    8) "i"
    9) "k"
   10) "o"
127.0.0.1:6379> scan 27
1) "0"                    # 返回0表示迭代结束
2) 1) "v"
   3) "x"
   4) "j"
   5) "p"
   6) "f"
   7) "n"
   8) "r"

  • 在上面这个例子中, 第一次迭代使用 0 作为游标, 表示开始一次新的迭代
  • 第二次迭代使用的是第一次迭代时返回的游标, 也即是命令回复第一个元素的值 1
  • 第三次迭代使用的是第一次迭代时返回的游标, 也即是命令回复第一个元素的值 27
  • 第三次迭代返回第一个元素为0 ,表示迭代已结束。

返回:

  • 反复执行scan命令同一个元素可能会被返回多次
  • 返回的元素个数最多可能会返回数十个元素
  • 返回两个元素的数组:第一个数组元素是用于进行下一次迭代的新游标;第二个是数组是包含了所有被迭代的元素
  • 返回0:表示迭代已经结束, 整个数据集已经被完整遍历过了
  • 从上操作可知:迭代命令并不保证每次执行都返回某个给定数量的元素
    • 甚至可能会返回零个元素, 但只要命令返回的游标不是 0 , 应用程序就不应该将迭代视作结束

MATCH 选项用法

SCAN cursor MATCH pattern:通过提供一个 glob 风格的模式参数, 让命令只返回和给定模式相匹配的元素, 这一点可以通过在执行增量式迭代命令时, 通过给定 MATCH <pattern> 参数来实现。

127.0.0.1:6379> mset a00 1 a001 1 a002 1 a003 1 a014 1 a015 1 a016 1
OK
127.0.0.1:6379> keys *
1) "a001"
2) "a01"
3) "a003"
4) "a014"
5) "a016"
6) "a015"
7) "a002"
8) "a00"
127.0.0.1:6379> scan 0 match *01  # 匹配01结尾的key
1) "0"
2) 1) "a001"
   2) "a01"
127.0.0.1:6379> scan 0 match a01* # 匹配a01开头的key
1) "0"
2) 1) "a016"
   2) "a015"
   3) "a01"
   4) "a014"

说明:

  • 命令第一步是先执行 scan cursor取出元素列表
  • 命令第二步才执行match操作
  • 所以返回值第二个数组有可能是空的

COUNT 选项用法

SCAN cursor COUNT count:增量式迭代命令不保证每次迭代所返回的元素数量, 但我们可以使用 COUNT 选项, 对命令的行为进行一定程度上的调整

  • 不使用count的情况下,COUNT 参数的默认值为 10
  • 该选项告知迭代命令, 在这次迭代中应该从数据集里返回多少元素
127.0.0.1:6379> mset a0111 1 a0121 1 a0031 1 a0041 1 a0051 1 a0061 1 a0071 1 a0081 1 a0011 1 a0101 1 a0111 1 a0121 1 a0131 1
OK
127.0.0.1:6379> mset a001 1 a002 1 a003 1 a004 1 a005 1 a006 1 a007 1 a008 1 a001 1 a010 1 a011 1 a012 1 a013 1
OK
127.0.0.1:6379> scan 0 count 5
1) "4"
2) 1) "a005"
   2) "a0041"
   3) "a0121"
   4) "a0031"
   5) "a004"
  • 在迭代一个足够大的集合键时, 如果用户没有使用 MATCH 选项, 那么命令返回的元素数量通常和 COUNT 选项指定的一样, 或者比 COUNT 选项指定的数量稍多一些
127.0.0.1:6379> scan 0 count 5
1) "6"
2) 1) "a22"         # count 5 返回了5个
   2) "a016"
   3) "a25"
   4) "a26"
   5) "a015"
127.0.0.1:6379> scan 0 count 8
1) "9"
2)  1) "a22"        # count 8 返回了10个
    2) "a016"
    3) "a25"
    4) "a26"
    5) "a015"
    6) "a21"
    7) "a001"
    8) "a01"
    9) "a003"
   10) "a014"
  • 在迭代一个编码为整数集合(intset,一个只由整数值构成的小集合)、 或者编码为压缩列表(ziplist,由不同值构成的一个小哈希或者一个小有序集合)时, 增量式迭代命令通常会无视 COUNT 选项指定的值
127.0.0.1:6379> flushdb
OK
127.0.0.1:6379> mset a001 1 a002 1 a003 1 a004 1 a005 1 a006 1 a007 1 a008 1 a001 1 a010 1 a011 1 a012 1 a013 1
127.0.0.1:6379> mset a0111 1 a0121 1 a0031 1 a0041 1 a0051 1 a0061 1 a0071 1 a0081 1 a0011 1 a0101 1 a0111 1 a0121 1 a0131 1

127.0.0.1:6379> scan 0 count 8
1) "2"
2)  1) "a005"    # 无似count 8限制,返回了10个
    2) "a0041"
    3) "a0121"
    4) "a0031"
    5) "a004"
    6) "a013"
    7) "a0011"
    8) "a0081"
    9) "a012"
   10) "a0061"

TYPE 选项用法

SCAN cursor TYPE type:type选择保证每次迭代输出均为指定的数据类型


127.0.0.1:6379> get a005
"1"
127.0.0.1:6379> type a005
string
127.0.0.1:6379> scan 0 type string
1) "2"
2)  1) "a005"
    2) "a0041"
    3) "a0121"
    4) "a0031"
    5) "a004"
    6) "a013"
    7) "a012"
    8) "a0081"
    9) "a0011"
   10) "a0061"
127.0.0.1:6379> scan 2 type string
1) "9"
2) 1) "a0101"
   2) "a010"
   3) "a008"
   4) "a007"
   5) "a0111"
   6) "a006"
   7) "a0071"
   8) "a001"
   9) "a003"

补充

并发执行多个迭代

在同一时间, 可以有任意多个客户端对同一数据集进行迭代, 客户端每次执行迭代都需要传入一个游标, 并在迭代执行之后获得一个新的游标, 而这个游标就包含了迭代的所有状态, 因此, 服务器无须为迭代记录任何状态。

中途停止迭代

因为迭代的所有状态都保存在游标里面, 而服务器无须为迭代保存任何状态, 所以客户端可以在中途停止一个迭代, 而无须对服务器进行任何通知。

即使有任意数量的迭代在中途停止, 也不会产生任何问题。

使用错误的游标进行增量式迭代

使用间断的(broken)、负数、超出范围或者其他非正常的游标来执行增量式迭代并不会造成服务器崩溃, 但可能会让命令产生未定义的行为。

未定义行为指的是, 增量式命令对返回值所做的保证可能会不再为真。

只有两种游标是合法的:

  • 在开始一个新的迭代时, 游标必须为 0 
  • 增量式迭代命令在执行之后返回的, 用于延续(continue)迭代过程的游标

迭代终结的保证

增量式迭代命令所使用的算法只保证在数据集的大小有界(bounded)的情况下, 迭代才会停止, 换句话说, 如果被迭代数据集的大小不断地增长的话, 增量式迭代命令可能永远也无法完成一次完整迭代。

从直觉上可以看出, 当一个数据集不断地变大时, 想要访问这个数据集中的所有元素就需要做越来越多的工作, 能否结束一个迭代取决于用户执行迭代的速度是否比数据集增长的速度更快。

时间复杂度:增量式迭代命令每次执行的复杂度为 O(1) , 对数据集进行一次完整迭代的复杂度为 O(N) , 其中 N 为数据集中的元素数量。

返回值:

  • SCAN 命令、 SSCAN 命令、 HSCAN 命令和 ZSCAN 命令都返回一个包含两个元素的回复: 回复的第一个元素是字符串表示的无符号 64 位整数(游标), 回复的第二个元素是包含了本次被迭代的元素
  • SCAN 命令返回的每个元素都是一个数据库键
  • SSCAN 命令返回的每个元素都是一个集合成员
  • HSCAN 命令返回的每个元素都是一个键值对,一个键值对由一个键和一个值组成
  • ZSCAN 命令返回的每个元素都是一个有序集合元素,一个有序集合元素由一个成员(member)和一个分值(score)组成

更多推荐

three.js 入门 初识

基本步骤:初始设置创建场景创建相机创建可见对象创建渲染器渲染场景安装npminstallthree引入import*asTHREEfrom"three";一、three三要素:场景、相机、渲染1.场景://创建场景constscene=newTHREE.Scene()2.相机:OrthographicCamera正交投

【前端设计模式】之工厂模式

工厂模式特性工厂模式是一种创建对象的设计模式,它通过使用工厂类来封装对象的创建逻辑,隐藏了具体对象的实例化过程。工厂模式的主要特性包括:封装对象的创建过程:工厂模式将对象的创建过程封装在一个工厂类中,客户端只需要通过调用工厂类的方法来获取所需的对象,而无需关心具体的实例化过程。提供统一的接口:工厂模式通常会定义一个统一

BIO、NIO、AIO有什么区别

什么是IOJava中I/O是以流为基础进行数据的输入输出的,所有数据被串行化(所谓串行化就是数据要按顺序进行输入输出)写入输出流。简单来说就是java通过io流方式和外部设备进行交互。在Java类库中,IO部分的内容是很庞大的,因为它涉及的领域很广泛:标准输入输出,文件的操作,网络上的数据传输流,字符串流,对象流等等等

基于招聘网站的大数据专业相关招聘信息建模与可视化分析

需要本项目的可以私信博主!!!在大数据时代背景下,数据积累导致大数据行业的人才需求快速上升,大量的招聘信息被发布在招聘平台上。深入研究这些信息能帮助相关人士更好地理解行业动态,并对其未来发展进行预测。本文主要通过分析51job网站上的大数据职位招聘信息,进行一次可视化的呈现。本研究首先使用Python爬虫技术,抓取51

Kubernetes中Pod的扩缩容介绍

Kubernetes中Pod的扩缩容介绍在实际生产系统中,我们经常会遇到某个服务需要扩容的场景,也可能会遇到由于资源紧张或者工作负载降低而需要减少服务实例数量的场景。此时可以利用Deployment/RC的Scale机制来完成这些工作。Kubernetes对Pod的扩缩容操作提供了手动和自动两种模式,手动模式通过执行k

零基础学前端(五)模仿QQ官网首页(重点练习HTML+CSS)

1.我前面已经将HTML和CSS讲解完成,希望初学者是跟着一步一步敲代码走过来的。2.我个人觉得先不着急进入Javascript的学习,要将前面写样式,写布局的基础打好,所以我编写了本篇模仿QQ网站首页,请认真阅读。一、确认目标我们要模仿qq官网的首页。你可以点击去看看官网的样子,我将目标图片先贴在下面。素材可以自己通

UWB学习——day4

UWB学习——day4技术劣势技术细节UWB频段系统调制方式UWB帧结构芯片实例技术劣势干扰其它技术,UWB技术目前允许在未授权的3.1GHz至10.6GHz频谱上运行,但该频谱上有许多其它无线通讯所在的频带,容易互相产生干扰,反而限制了适用范围。时钟同步要求高,测距和定位需要高精度的时钟同步频谱利用率低,虽然拥有巨大

实现不同局域网间的文件共享和端口映射,使用Python自带的HTTP服务

文章目录1.前言2.本地文件服务器搭建2.1python的安装和设置2.2cpolar的安装和注册3.本地文件服务器的发布3.1Cpolar云端设置3.2Cpolar本地设置4.公网访问测试5.结语1.前言数据共享作为和连接作为互联网的基础应用,不仅在商业和办公场景有广泛的应用,对于个人用户也有很强的实用意义。也正因如

产业互联网,并非消费互联网式的置身事外

在消费互联网时代,我们听到最多的一个词汇,便是「收割」。当「收割」发展到一定的程度,我们还听到了一个词,叫「豢养」。不得不说,在流量充沛的年代里,通过「收割」和「豢养」的方式,的确是可以获得一定的发展潜能的。然而,等到流量的红利不再,通过简单粗暴的「收割」和「豢养」开始表现出来了越来越多的问题和弊端。我们看到的流量的沉

HEXO 基本使用

1新建、编辑并预览文章1.新建文章hexonew[layout]title#或hexon[layout]title创建文章前要先选定模板,在hexo中也叫做布局。hexo支持三种布局(layout):post(默认)、draft、page。我们先介绍如何使用已有布局,后面还将会介绍如何自定义布局。在博客目录下输入以下命

macOS 中 聚焦搜索 的使用教程

macOS中的聚焦搜索是一个强大的工具,它可以帮助你快速找到文件、应用程序、联系人、电子邮件、互联网搜索结果等。下面是macOS中聚焦搜索的使用教程:1.打开聚焦搜索:使用键盘快捷键:按下键盘上的Command键和空格键(⌘+空格),即可打开聚焦搜索。这是最快的方式。使用鼠标或触控板:点击屏幕右上角的聚焦搜索图标(放大

热文推荐