Redis - list 列表

前言

        列表类似于 Java 中的数组或者顺序表,在 Redis 中,可以对列表两端插⼊(push)和弹出(pop),还可以获取指定范围的元素列表、 获取指定索引下标的元素等。列表是⼀种⽐较灵活的数据结构,它可以充当栈和队列的⻆⾊,在实际开发上有很多应⽤场景。
        注意:list 内部的编码方式并不是一个简单的数组,更接近于双端队列

列表两端插入和弹出操作

        约定最左侧下标为 0 ,redis 的下标支持负数下标,最右侧是 -1

列表类型的特点

a.列表中的元素是有序的

        " 有序 " 的含义要根据上下文区分,有的时候谈到有序指的是升序降序,而这里谈到的有序是指顺序很关键,列表有下标,如果把两个下标之间的值进行交换,得到的列表和之前的列表就不相同

b.列表中的元素是允许重复的

c.可以当作栈和队列来使用

        因为列表的头和尾都可以高效的插入和删除元素,所以可以做出相应的限制,当作栈和队列来使用

命令

LPUSH 将⼀个或者多个元素从左侧放入(头插)到 list 中

语法

LPUSH key element [element ...]

时间复杂度:只插⼊⼀个元素为 O(1), 插⼊多个元素为 O(N), N 为插⼊元素个数.

返回值:插⼊后 list 的⻓度

        通过上图的操作,key1 对应的列表中的数据为 4 3 2 1

        注意:如果 key 已经存在,并且 key 对应的 value 类型不是 list,那么就会报错(redis 中各种数据类型的操作都是类似的效果)

LPUSHX 在 key 存在时,将⼀个或者多个元素从左侧放入(头插)到 list 中

语法

LPUSHX key element [element ...]

时间复杂度:只插⼊⼀个元素为 O(1), 插⼊多个元素为 O(N), N 为插⼊元素个数.

返回值:插⼊后 list 的⻓度。

        上图中通过 lpushx 插入数据到列表中得到的返回值是 0 就说明了没有成功插入数据,因为该列表当前不存在

RPUSH 将⼀个或者多个元素从右侧放⼊(尾插)到 list 中

语法

RPUSH key element [element ...]

时间复杂度:只插⼊⼀个元素为 O(1), 插⼊多个元素为 O(N), N 为插⼊元素个数

返回值:插⼊后 list 的⻓度

        通过上图的操作,key 对应的列表中的数据为 1 2 3 4 5

RPUSHX 在 key 存在时,将⼀个或者多个元素从右侧放入(尾插)到 list 中

语法

RPUSHX key element [element ...]

时间复杂度:只插⼊⼀个元素为 O(1), 插⼊多个元素为 O(N), N 为插⼊元素个数.

返回值:插⼊后 list 的⻓度

LRANGE 获取指定区间的所有元素

        获取从 start 到 stop 区间的所有元素,左闭右闭。下标可以使用负数

语法

LRANGE key start stop

时间复杂度:O(N)

返回值:指定区间的元素

        -1 代表最后一个元素,所以获取 0 ~ -1 的元素相当于获取列表中所有的元素

        谈到下标,往往会关注超出范围的情况,在 Redis 中,如果给定的区间非法,比如超出下标,会尽可能的获取能获取到的内容,程序的容错能力很强

        如图,要获取 0 ~ 100 区间内的数据,而列表中只有 5 个数据,很明显 100 超出了列表的下标,但 Redis 并没有报错,而是尽可能的获取了0 ~ 100 中能够获取到的数据

LPOP 从 list 左侧取出元素(即头删)

语法

LPOP key

时间复杂度:O(1) 

返回值:取出的元素或者 nil

RPOP 从 list 右侧取出元素(即尾删)

语法

RPOP key

时间复杂度:O(1) 

返回值:取出的元素或者 nil

LINDEX 获取从左数 index 位置的元素

语法

LINDEX key index

时间复杂度:O(N) 

        应该会有读者疑惑,通过下标获取元素时间复杂度不应该是 O(1) 吗,实际上 Redis 的 list(列表)类型的内部编码不是一个普通的顺序表,所以不能按照普通的顺序表来考虑该列表

返回值:取出的元素或者 nil

LINSERT 在特定位置插入元素

语法

LINSERT key <BEFORE | AFTER> pivot element

        pivot 代表要插入数据的基准值,element 代表插入的数据

时间复杂度:O(N)

返回值:插⼊后的 list ⻓度

        万一基准值存在多个怎么办?数据要如何插入?

        如上图,当基准值 2 存在多个时,选取从左到右的第一个 2 作为基准值

LLEN 获取 list 长度

语法

 LLEN key

时间复杂度:O(1)

        获取 list 的长度并不需要遍历列表中的内容,因为 redis 为列表维护了一个变量来存储列表的长度,当要获取列表长度时,直接去获取该变量即可

返回值:list 的⻓度

BLPOP 从 list 左侧取出元素(即头删)

        LPOP 的阻塞版本

        • 在列表中有元素的情况下,阻塞和⾮阻塞表现是⼀致的。但如果列表中没有元素,⾮阻塞版本会直接返回 nil,但阻塞版本会根据 timeout,阻塞⼀段时间,期间 Redis 可以执⾏其他命令,但要求执⾏该命令的客户端会表现为阻塞状态

        • 命令中如果设置了多个键,那么会从左向右进⾏遍历键,⼀旦有⼀个键对应的列表中可以弹出元 素,命令⽴即返回。

        • 如果多个客户端同时对⼀个键执⾏ pop,则最先执⾏命令的客户端会得到弹出的元素

语法

BLPOP key [key ...] timeout

        列表 key 可以有多个,表示可以同时监控多个列表,哪个列表有数据就取哪个列表

        timeout 是阻塞的时间,阻塞时间过了以后还没有获取到数据就返回 nil

时间复杂度:O(1)

返回值:取出的元素或者 nil

BRPOP 从 list 右侧取出元素(即尾删)

RPOP 的阻塞版本(详细内容参考对 BLPOP 的介绍)

语法

BRPOP key [key ...] timeout

时间复杂度:O(1)

返回值:取出的元素或者 nil

内部编码

以前列表类型的内部编码有两种:

• ziplist(压缩列表):当列表的元素个数⼩于 list-max-ziplist-entries 配置(默认 512 个),同时 列表中每个元素的⻓度都⼩于 list-max-ziplist-value 配置(默认 64 字节)时,Redis 会选⽤  ziplist 来作为列表的内部编码实现来减少内存消耗

• linkedlist(链表):当列表类型⽆法满⾜ ziplist 的条件时,Redis 会使⽤ linkedlist 作为列表的内 部实现

        因为 ziplist 编码方式虽然节省空间,但是当元素个数较多或元素较长时会大大影响存取数据的效率,所以就需要改变编码方式为 linkedlist

        注意:上述的两种编码方式我特意加上了以前,因为现在 list(列表)的内部编码方式作出了改进,现在的内部编码方式是 quicklist ,quicklist 是 ziplist(压缩列表)和 linkedlist(链表)的结合,整体还是个链表,但链表的每个节点是一个压缩列表

        通过这种方式可以节省空间,并且压缩列表不会太长,不会明显的影响读取数据的效率

使用场景

消息队列

        Redis 可以使⽤ lpush + brpop 命令组合实现经典的阻塞式 ⽣产者-消费者 模型队列, ⽣产者客户端使⽤ lpush 从列表左侧插⼊元素,多个消费者客户端使⽤ brpop 命令阻塞式地从队列中  "争抢" 队⾸元素。通过多个客户端来保证消费的负载均衡和⾼可⽤性

        有读者可能会疑惑,消息队列的存取元素不是都应该实现阻塞吗?但 lpush + brpop 命令组合明显只实现取元素的阻塞,实际上对于 Redis 的数据类型是几乎不会出现存满的情况的,首先如果 Redis 的一个数据类型中存了太多的元素,那么对其进行操作时很容易发生阻塞,所以程序员不会让 Redis 的一个数据类型中存了太多的元素,而且 list 中最多存多少元素是可以配置的,所以通常情况下只会出现列表中没有数据的情况,不会出现存满的情况。

 Redis 阻塞消息队列模型

        哪个消费者先执行 brpop 命令,哪个消费者就先获得列表中的数据

Redis 分频道阻塞消息队列模型

        消费者可以同时监视多个列表,获取数据

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

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

相关文章

Loss-Attention

辅助信息 作者不开issue了&#xff0c;不建议复现

学习C++是否有必要学习Boost库?

C作为一门强大且灵活的编程语言&#xff0c;在软件开发领域有着广泛的应用。而在C的学习过程中&#xff0c;Boost库是一个经常被提及的重要资源。那么&#xff0c;对于C的学习者而言&#xff0c;是否有必要投入精力去学习Boost库呢&#xff1f;本文将就此问题展开详尽讨论。 一…

面试题:Java中的类加载器

1. 什么是类加载器&#xff0c;类加载器有哪些? 要想理解类加载器的话&#xff0c;务必要先清楚对于一个Java文件&#xff0c;它从编译到执行的整个过程。 类加载器&#xff1a;用于装载字节码文件(.class文件)运行时数据区&#xff1a;用于分配存储空间执行引擎&#xff1a;…

零代码编程:用kimichat批量重命名txt文本文件

一个文件夹中有很多个txt文本文件&#xff0c;需要全部进行重命名。 可以在kimichat中输入提示词&#xff1a; 你是一个Python编程专家&#xff0c;要完成一个关于批量重命名txt文本文件的Python脚本&#xff0c;下面是具体步骤&#xff1a; D:\Best Seller Books 这个文件夹…

chatGPT中文在线版本(亲测可用

ChatGPT是一个先进的自然语言处理模型&#xff0c;由OpenAI开发。它通过深度学习技术训练而成&#xff0c;可以进行对话、回答问题等多种自然语言处理任务。对于学生、开发者、研究人员和任何对人工智能感兴趣的人来说&#xff0c;这是一个非常有用的工具。 最近找到一个国内可…

【算法篇】逐步理解动态规划1(斐波那契数列模型)

目录 斐波那契数列模型 1. 第N个泰波那契数 2.使用最小花费爬楼梯 3.解码方法 学过算法的应该知道&#xff0c;动态规划一直都是一个非常难的模块&#xff0c;无论是状态转移方程的定义还是dp表的填表&#xff0c;都非常难找到思路。在这个算法的支线专题中我会结合很多力…

html网页制作-3D旋转卡片

网页制作-3D旋转卡片 两种旋转卡片的制作 旋转卡片&#xff08;1&#xff09; 代码 html代码 <!DOCTYPE html> <html lang"en"> <head><meta charset"UTF-8"><meta name"viewport" content"widthdevice-wid…

遍历目录下的某个文件并删除

目录 需求 编写过程 演示 需求 大家在学习时可能会有一个自己的小目录&#xff0c;里面放着各种奇葩代码&#xff0c;有天突然发现&#xff0c;没有空间了&#xff0c;这时候发现遗留了很多的可执行文件&#xff0c;大大的浪费了我们的空间&#xff0c;但是由于层数深&#…

汽车ABS的bangbang控制和模糊PID控制

1、内容简介 略 82-可以交流、咨询、答疑 2、内容说明 摘要&#xff1a;本文旨在设计一种利用模糊控制理论优化的pid控制器&#xff0c;控制abs系统&#xff0c;达到对滑移率最佳控制范围的要求 &#xff0c;所提出的方案采用级联控制架构&#xff1a;设计用于外环中的车轮打…

第四百二十二回

文章目录 1. 概念介绍2. 思路与方法2.1 实现思路2.2 实现方法 3. 示例代码4. 内容总结 我们在上一章回中介绍了"自定义标题栏"相关的内容&#xff0c;本章回中将介绍自定义Action菜单.闲话休提&#xff0c;让我们一起Talk Flutter吧。 1. 概念介绍 我们在这里提到的…

基于大数据的空气质量预测和可视化分析

城市空气质量数据采集系统设计与实现 &#x1f3d9;️ 研究背景 &#x1f32c;️ 城市化与环境挑战&#xff1a;随着城市化进程的加快&#xff0c;环境污染问题&#xff0c;尤其是空气质量问题&#xff0c;已成为公众关注的焦点。数据监测的重要性&#xff1a;城市空气质量数…

哪本书最了解孩子?跟《米小圈上学记》一起做孩子的“引路人”!

孩子是发展中的人&#xff0c;需要家长的陪伴&#xff0c;孩子的身心发展是有规侓的&#xff0c;是处于发展过程中的人。我们要学会尊重孩子的发展&#xff0c;从兴趣出发&#xff0c;关注孩子的成长。但是&#xff0c;家长不可能无时无刻都能陪在孩子身边&#xff0c;他需要一…