【Redis】zset常用命令集合间操作内部编码使用场景

文章目录

  • 前置知识
    • 列表、集合、有序集合三者的异同点
  • 普通命令
    • ZADD
    • ZCARD
    • ZCOUNT
    • ZRANGE
    • ZREVRANGE
    • ZRANGEBYSCORE
    • ZPOPMAX
    • BZPOPMAX
    • ZPOPMIN
    • BZPOPMIN
    • ZRANK
    • ZREVRANK
    • ZSCORE
    • ZREM
    • ZREMRANGEBYRANK
    • ZREMRANGEBYSCORE
    • ZINCRBY
  • 集合之间的操作
    • ZINTERSTORE
    • ZUNIONSTORE
  • 命令小结
  • 内部编码
    • 测试内部编码
  • 使用场景

前置知识

有序集合中的每个元素都有⼀个唯⼀的浮点类型的分数(score)与之关联,这使得有序集合中的元素是可以维护有序性的,进行排序的时候,就是按照此处的分数大小进行升序/降序排序

注意:zset主要还是用来存member,score只是辅助

例子:使用有序集合显⽰三国中的武将的武⼒

image-20231021213941478
  • 分数不同,则按照分数来升序排序(zset内部按照升序排列),分数相同的时候,再按照元素自身字符串的字典序来排序

有序集合中的元素是不能重复的,但分数允许重复。类⽐于⼀次考试之后,每个⼈⼀定有⼀个唯⼀的分数,但分数允许相同


列表、集合、有序集合三者的异同点

image-20231021214034003


普通命令

ZADD

添加或者更新指定的元素以及关联的分数到zset中,分数应该符合double类型,+inf/-inf作为正负极限也是合法的

语法:ZADD key [NX | XX] [GT | LT] [CH] [INCR] score member [score member...]

相关选项:

  • XX:仅仅⽤于更新已经存在的元素,不会添加新元素
  • NX:仅⽤于添加新元素,不会更新已经存在的元素
  • CH:默认情况下,ZADD返回的是本次添加的元素个数,但指定这个选项之后,就会还包含本次更新的元素的个数,可能会影响zadd的返回值
    • 注意:通常ZADD的返回值只计算添加的新元素的数量。
  • INCR:此时命令类似ZINCRBY的效果,将元素的分数加上指定的分数。此时只能指定⼀个元素和分数

返回值:本次添加成功的元素个数 时间复杂度:O(log(N))

  • 注意:之前hash,set,list添加一个元素都是 O ( 1 ) O(1) O(1),但是zset添加元素的时间复杂度为 O ( l o g N ) O(logN) O(logN),这是因为zset是有序结构,要新增新元素,要放到合适的位置上,之所以不是 O ( N ) O(N) O(N),是因为zset内部的数据结构为调表

image-20231023205409741

不加NX || XX选项的时候:

  • 如果当前member不存在,此时就会达到添加新member的效果
  • 如果当前member已经存在,此时就会更新分数

image-20231026204150134

XX:只更新已经存在的元素。不要添加新元素。

NX:只添加新元素。不要更新已经存在的元素。


LT和ET:

image-20231026204212618

LT:现在要更新分数了,发现现在给定的新的分数比之前的分数小,此时就更新成功,否则就不更新

  • 只有当新分数小于当前分数时才会更新现有元素。这个标志不会阻止添加新元素

GT:仅在新分数大于当前分数时更新现有元素。这个标志不会阻止添加新元素。


ZCARD

获取⼀个zset的基数(cardinality),即zset中的元素个数

语法:ZCARD key

返回值:zset内的元素个数 时间复杂度:O(1)

image-20231023205457393

ZCOUNT

返回分数在min和max之间的元素个数,默认情况下,min和max都是包含的,即范围是 [ m i n , m a x ] [min,max] [minmax],可以通过(来排除边界值

语法:ZCOUNT key min max

返回值:满⾜条件的元素列表个数 时间复杂度:O(log(N))

image-20231023205653093

时间复杂度分析:先根据min找到对应的元素,再根据max找到对应的元素(时间复杂度为: O ( l o g N ) O(logN) O(logN)),实际上,zset内部会记录每个元素当前的排行/次序,查询到元素就直接知道了元素所在的次序(下标),就可以直接把max对应的元素的次序和min对应的元素的次序做减法即可

注意1:min和max可以写成浮点数,因为zset的分数本身就是浮点数

注意2:在浮点数当中,存在两个特殊的数值:inf表示无穷大,-inf表示负无穷大(负无穷大 != 无穷小)


ZRANGE

返回指定区间⾥的元素,分数按照升序。带上WITHSCORES可以把分数也返回

语法:ZRANGE key start stop [WITHSCORES] 
  • 注意:此处的[start,stop]为下标构成的区间.从0开始,⽀持负数.

返回值:区间内的元素列表 时间复杂度:O(log(N)+M)

image-20231023205739557

时间复杂度分析:先根据下标找到边界值(O(logN)),然后从start对应位置开始往后遍历,M:start-stop区间的元素个数

ZREVRANGE

返回指定区间⾥的元素,按照分数降序打印,带上WITHSCORES可以把分数也返回

语法:ZREVRANGE key start stop [WITHSCORES]

返回值:区间内的元素列表 时间复杂度:O(log(N)+M)

image-20231024112139729

ZRANGEBYSCORE

返回分数在min和max之间的元素,默认情况下,min和max都是包含的,可以通过(排除

语法:ZRANGEBYSCORE key min max [WITHSCORES] 

返回值:区间内的元素列表 时间复杂度:O(log(N)+M)

image-20231024112305645

ZPOPMAX

删除并返回分数最⾼的count个元素

语法:ZPOPMAX key [count] 

返回值:分数和元素列表 时间复杂度:*O(log(N)M)

  • N:有序集合元素个数 M:count要删除的元素个数

image-20231024112408261

如果存在多个元素分数相同,同时为最大值的时候,zpopmax进行删除,如果不指定count,仍然只删除其中一个元素

  • 如果分数相同会按照member字符串的字典序排序

注意:此处删除的是最大值,在有序集合当中,最大值相当于就是最后一个元素(删除最大值==>尾删),既然是尾删,此时可以把最后一个元素的位置特殊记录下来,后续就可以以 O ( 1 ) O(1) O(1)的复杂度进行删除,所以 O ( l o g N ) = > O ( 1 ) O(logN) =>O(1) O(logN)=>O(1)是可能的,但是redis并没有这么做


BZPOPMAX

ZPOPMAX的阻塞版本,有序集合可以视为是优先级队列,此时使用bzpopmax就相当于是一个带有阻塞功能的优先级队列

语法:BZPOPMAX key [key ...] timeout #timeout:超时时间,单位是s,支持小数形式,写做0.1 就是代表100ms

返回值:元素列表 时间复杂度:O(log(N)) =>删除最大值花费的时间

image-20231024112512538

可以同时等待多个key对应的有序集合当中的元素就绪,阻塞到有其它客户端往任意一个key当中插入元素。如果有序集合已经有元素了,直接就能返回,不会阻塞

ZPOPMIN

删除并返回有序集合当中分数最低的count个元素

语法:ZPOPMIN key [count] 

返回值:分数和元素列表 时间复杂度:O(log(N)*M)

image-20231024112657296

BZPOPMIN

ZPOPMIN的阻塞版本

语法:BZPOPMIN key [key ...] timeout 

返回值:元素列表 时间复杂度:O(log(N))

image-20231024112738424

ZRANK

返回指定元素的排名,升序计算下标

语法:ZRANK key member 

返回值:排名 时间复杂度:O(log(N)) =>查询位置的过程

image-20231024112956618

注意:zcount在计算的时候,就是先根据分数找到元素,再根据元素获取到排名,再把排名相减就得到了元素个数

zrank得到的下标,是从前往后计算的,下标从0开始

ZREVRANK

返回指定元素的排名,降序计算下标(是从后往前计算的,下标从0开始)

语法:ZREVRANK key member 

返回值:排名 时间复杂度:O(log(N))

image-20231024113017435


ZSCORE

返回指定元素的分数

语法:ZSCORE key member 

返回值:分数 时间复杂度:O(1)

image-20231024113034949

注意:前面根据member找分数都是 l o g N logN logN,此处相当于是redis对于这样的查询做了优化,付出了额外的空间代价,优化到了 O ( 1 ) O(1) O(1)查询

ZREM

删除指定的元素

语法:ZREM key member [member ...] 

返回值:本次操作删除的元素个数 时间复杂度:O(M*log(N))

  • N:有序集合元素个数 M:参数当中member的个数

image-20231024113052529

ZREMRANGEBYRANK

按照排序,升序删除指定范围的元素,左闭右闭 [ s t a r t , s t o p ] [start,stop] [start,stop]

语法:ZREMRANGEBYRANK key start stop 

返回值:本次操作删除的元素个数 时间复杂度:O(log(N)+M)

  • N:有序集合元素个数 M: s t o p − s t a r t stop- start stopstart区间的元素个数

image-20231024113502072

ZREMRANGEBYSCORE

按照分数删除指定范围的元素,左闭右闭 [ m i n , m a ] [min,ma] [min,ma],可以通过(排除边界值

语法:ZREMRANGEBYSCORE key min max 

返回值:本次操作删除的元素个数 时间复杂度:O(log(N)+M)

image-20231024113618390

ZINCRBY

为指定的元素的关联分数添加指定的分数值,修改之后,仍然会保持整个有序集合是升序的

语法:ZINCRBY key increment member 

返回值:增加后元素的分数 时间复杂度:O(log(N))

image-20231024113716214


集合之间的操作

ZINTERSTORE

出给定有序集合中元素的交集并保存进⽬标有序集合中,在合并过程中以元素为单位进⾏合并,元素对应的分数按照不同的聚合⽅式和权重得到新的分数

语法:ZINTERSTORE destination numkeys key [key ...] [WEIGHTS weight  [weight ...]] [AGGREGATE <SUM | MIN | MAX>]
  • destination:要把求得的交集存储到哪个key当中,对应的zset
  • numkeys:是一个整数,描述后续有几个key参与交集运算
  • weights:权重,因为是有序集合,带有分数,此处的权重相当于是一个系数,会乘当前的分数
  • AGGREGATE:表示最后交集的结果取什么结果,求和,最小值,最大值

有序集合的交集操作

image-20231021222217546

返回值:⽬标集合中的元素个数

时间复杂度:O(N*K)+O(M*log(M))

  • N是输⼊的有序集合中,最⼩的有序集合的元素个数
  • K是输⼊了⼏个有序集合 ==>多少个有序集合进行合并求交集
  • M是最终结果的有序集合的元素个数

K一般不会很多,可以近似看成1,也可以认为N和M是接近的(同一个数量级),那么O(N*K)+O(M*log(M)) =>O(M) + O(M * logM) =>O(M*logM)

image-20231024114600105


ZUNIONSTORE

求出给定有序集合中元素的并集并保存进⽬标有序集合中,在合并过程中以元素为单位进⾏合并,元素对应的分数按照不同的聚合⽅式和权重得到新的分数

语法:ZUNIONSTORE destination numkeys key [key ...] [WEIGHTS weight  [weight ...]] [AGGREGATE <SUM | MIN | MAX>]

有序集合的并集操作

image-20231021222423802


返回值:⽬标集合中的元素个数

时间复杂度:O(N)+O(M*log(M))N是输⼊的有序集合总的元素个数;M是最终结果的有序集合的元素个数


命令小结

image-20231021222600898


内部编码

有序集合类型的内部编码有两种:

  • ziplist(压缩列表):当有序集合的元素个数⼩于zset-max-ziplist-entries配置(默认128个),同时每个元素的值都⼩于zset-max-ziplist-value配置(默认64字节)时,Redis会⽤ziplist来作为有序集合的内部实现,ziplist可以有效减少内存的使⽤
  • skiplist(跳表):当ziplist条件不满⾜时(元素个数比较多或者单个元素比较大),有序集合会使⽤skiplist作为内部实现,因为此时ziplist的操作效率会下降
    • 跳表是一个复杂链表,查询元素的时间复杂度为 l o g N logN logN,相比于树形结构,更适合按照范围获取元素

测试内部编码

1)当元素个数较少且每个元素较⼩时,内部编码为ziplist

image-20231021222747827

2)当元素个数超过128个,内部编码skiplist

3)当某个元素⼤于64字节时,内部编码skiplist

image-20231021222822679


使用场景

有序集合⽐较典型的使⽤场景就是排⾏榜系统。例如常⻅的⽹站上的热榜信息,榜单的维度可能是多⽅⾯的:按照时间、按照阅读量、按照点赞量。

例子:使⽤点赞数这个维度,维护每天的热榜:

1)添加⽤⼾赞数:例如⽤⼾james发布了⼀篇⽂章,并获得3个赞,可以使⽤有序集合的zadd和zincrby功能

zadd user:ranking:2022-03-15 3 james 

之后如果再获得赞,可以使⽤zincrby:

zincrby user:ranking:2022-03-15 1 james 

2)取消⽤⼾赞数:由于各种原因(例如⽤⼾注销、⽤⼾作弊等)需要将⽤⼾删除,此时需要将⽤⼾从榜单中删除掉,可以使⽤zrem。例如删除成员tom

zrem user:ranking:2022-03-15 tom 

3)展⽰获取赞数最多的10个⽤⼾

zrevrangebyrank user:ranking:2022-03-15 0 9

4)展⽰⽤⼾信息以及⽤⼾分数

该功能将⽤⼾名作为键后缀,将⽤⼾信息保存在哈希类型中,⾄于⽤⼾的分数和排名可以使⽤zscore和zrank来实现

hgetall user:info:tom
zscore user:ranking:2022-03-15 mike
zrank user:ranking:2022-03-15 mik

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/148226.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

C/C++---------------LeetCode第LCR. 024.反转链表

反转链表 题目及要求双指针 题目及要求 双指针 思路&#xff1a;遍历链表&#xff0c;并在访问各节点时修改 next 引用指向&#xff0c;首先&#xff0c;检查链表是否为空或者只有一个节点&#xff0c;如果是的话直接返回原始的头节点&#xff0c;然后使用三个指针来迭代整个…

windows 安装 Oracle Database 19c

目录 什么是 Oracle 数据库 下载 Oracle 数据库 解压文件 运行安装程序 测试连接 什么是 Oracle 数据库 Oracle数据库是由美国Oracle Corporation&#xff08;甲骨文公司&#xff09;开发和提供的一种关系型数据库管理系统&#xff0c;它是一种强大的关系型数据库管理系统…

lambda表达式c++

介绍 可调用对象 对于一个表达式&#xff0c;如果可以对其使用调用运算符&#xff08;&#xff09;&#xff0c;则称它为可调用对象。如函数就是一个可调用对象&#xff0c;当我们定义了一个函数f(int)时&#xff0c;我们可以通过f(5)来调用它。 可调用对象有&#xff1a; …

基于springboot实现校园在线拍卖系统项目【项目源码】计算机毕业设计

基于springboot实现校园在线拍卖系统演示 Javar技术 JavaScript是一种网络脚本语言&#xff0c;广泛运用于web应用开发&#xff0c;可以用来添加网页的格式动态效果&#xff0c;该语言不用进行预编译就直接运行&#xff0c;可以直接嵌入HTML语言中&#xff0c;写成js语言&…

Java多线程核心技术第一阶段-Java多线程基础 02

接上篇&#xff1a;Java多线程核心技术第一阶段-Java多线程基础 01 3.3 清除中断状态的使用场景 this.interrupted()方法具有清除状态标志值的功能&#xff0c;借用此特性可以实现一些效果。 【示例3.3.1】在MyThread4线程中向list1和list2存放数据&#xff0c;基于单一职责原…

asp.net mvc点餐系统餐厅管理系统

1. 主要功能 ① 管理员、收银员、厨师的登录 ② 管理员查看、添加、删除菜品类型 ③ 管理员查看、添加、删除菜品&#xff0c;对菜品信息进行简介和封面的修改 ④ 收银员浏览、搜索菜品&#xff0c;加入购物车后进行结算&#xff0c;生成订单 ⑤ 厨师查看待完成菜品信息…

开拓经验专栏:从十来天的晨型人体验开始

文章目录 拓新缘起契机实践心得 拓新 确定要新开一个板块&#xff0c;用来记录持续自我提升的经验和教训&#xff0c;着实遭遇了不少阻力。 首先&#xff0c;我的语文功底一向不行&#xff0c;当年高考前&#xff0c;语文分数在及格线上下跳动都是常事&#xff0c;现在却要通…

DAC实验(DAC 输出三角波实验)(DAC 输出正弦波实验)

DAC 输出三角波实验 本实验我们来学习使用如何让 DAC 输出三角波&#xff0c;DAC 初始化部分还是用 DAC 输出实验 的&#xff0c;所以做本实验的前提是先学习 DAC 输出实验。 使用 DAC 输出三角波&#xff0c;通过 KEY0/KEY1 两个按键&#xff0c;控制 DAC1 的通道 1 输出两种…

LinkWeChat V4.9.8 版本发布

LinkWeChat v4.9.8 已经发布&#xff0c;基于企业微信的 SCRM 系统 LinkWeChat 是国内首个基于企业微信的开源 SCRM&#xff0c;在集成了企微强大的开放能力的基础上&#xff0c;进一步升级拓展灵活高效的客户运营能力及多元化精准营销能力&#xff0c;让客户与企业之间建立强…

Nginx反向代理和负载均衡

1.反向代理 反向代理&#xff08;Reverse Proxy&#xff09;方式是指以代理服务器来接受internet上的连接请求&#xff0c;然后将请求转发给内部网络上的服务器&#xff0c;并将从服务器上得到的结果返回给internet上请求连接的客户端&#xff0c;此时代理服务器对外就表现为一…

记录:RK3568显示异常。

最近调一个RK3568的新板子&#xff0c;板子其它接口功能都调试ok。可唯独在适配显示时发现&#xff0c;HDMI和MIPI显示均出现异常。当系统启动要进入桌面时候内核就开始报错。 因为这套源码之前在其它的板子上适配过&#xff0c;所以第一反应就是硬件问题或者是那个电压没配置…

SQL注入1

对sql进行一个小结 还有其他的注入 其他注入:堆叠注入&#xff0c;宽字节注入&#xff0c;二次注入 首先是数值和字符 id1 and 11和id1 and 12 如果这两个语句返回的页面不一样就说明是数字型 id1 and 11#和id1 and 12# 如果这两个语句返回的页面不一样就说明是字符型 常…

【Promise12数据集】Promise12数据集介绍和预处理

【Segment Anything Model】做分割的专栏链接&#xff0c;欢迎来学习。 【博主微信】cvxiayixiao 本专栏为公开数据集的介绍和预处理&#xff0c;持续更新中。 要是只想把Promise12数据集的raw形式分割为png形式&#xff0c;快速导航&#xff0c;直接看2&#xff0c;4标题即可 …

【精选】项目管理工具——Maven详解

Maven简介 Maven是一个项目管理工具。它可以帮助程序员构建工程&#xff0c;管理jar包&#xff0c;编译代码&#xff0c;完成测试&#xff0c;项目打包等等。 Maven工具是基于POM&#xff08;Project Object Model&#xff0c;项目对象模型&#xff09;实现的。在Maven的管理下…

Spring Framework 6.1 正式 GA

Spring Framework 6.1在运行时方面针对 JDK 21 和 Jakarta EE 10 上提供了一级支持&#xff0c;同时保留了 JDK 17 和 Jakarta EE 9 基线。Spring 还通过精细的元数据推理跟踪 GraalVM for JDK 21 的演变&#xff0c;同时暂时保持与 GraalVM 22.3 的兼容性。 主要变化 支持 JD…

Unity在Windows选项下没有Auto Streaming

Unity在Windows选项下没有Auto Streaming Unity Auto Streaming插件按网上说的不太好使最终解决方案 Unity Auto Streaming插件 我用的版本是个人版免费版&#xff0c;版本号是&#xff1a;2021.2.5f1c1&#xff0c;我的里边Windows下看不到Auto Streaming选项,就像下边这张图…

Python-pptx教程之二操作已有PPT模板文件

文章目录 简单的案例找到要修改的元素修改幻灯片中的文本代码使用示例 修改幻灯片的图片代码使用示例 删除幻灯片代码使用示例 获取PPT中所有的文本内容获取PPT中所有的图片总结 在上一篇中我们已经学会了如何从零开始生成PPT文件&#xff0c;从零开始生成较为复杂的PPT是非常消…

Jmeter——循环控制器中实现Counter计数器的次数重置

近期在使用Jmeter编写个辅助测试的脚本&#xff0c;用到了多个Loop Controller和Counter。 当时想的思路就是三个可变的数量值&#xff0c;使用循环实现&#xff1b;但第三个可变值的数量次数&#xff0c;是基于第二次循环中得到的结果才能确认最终次数&#xff0c;每次的结果…

爱奇艺大数据离在线混部

混部作为一种提高资源利用率、降低成本的的方案&#xff0c;被业界普遍认可。爱奇艺在云原生化与降本增效的过程中&#xff0c;成功将大数据离线计算、音视频内容处理等工作负载与在线业务进行了混部&#xff0c;并且取得了阶段性收益。本文重点以大数据为例&#xff0c;介绍从…

HDFS、MapReduce原理--学习笔记

1.Hadoop框架 1.1框架与Hadoop架构简介 &#xff08;1&#xff09;广义解释 从广义上来说&#xff0c;随着大数据开发技术的快速发展与逐步成熟&#xff0c;在行业里&#xff0c;Hadoop可以泛指为&#xff1a;Hadoop生态圈。 也就是说&#xff0c;Hadoop指的是大数据生态圈整…