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.mzph.cn/news/768269.shtml

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

相关文章

(一)基于IDEA的JAVA基础7

关系运算符 运算符 含义 范例 结果 等于 12 false &#xff01; 不等于 1&#xff01;2 true > 大于 1>2 false < 小于 …

微服务(基础篇-001-介绍、Eureka)

目录 认识微服务&#xff08;1&#xff09; 服务架构演变&#xff08;1.1&#xff09; 单体架构&#xff08;1.1.1&#xff09; 分布式架构&#xff08;1.1.2&#xff09; 微服务&#xff08;1.1.3&#xff09; 微服务结构 微服务技术对比 企业需求 SpringCloud(1.2) …

思科网络中DHCP中继的配置

一、什么是DHCP中继&#xff1f;DHCP中继有什么用? &#xff08;1&#xff09;DHCP中继是指一种网络设备或服务&#xff0c;用于在不同的子网之间传递DHCP&#xff08;动态主机配置协议&#xff09;消息。DHCP中继的作用是帮助客户端设备获取IP地址和其他网络配置信息&#x…

jvm底层

逐步细化 静态链接&#xff1a;静态方法(符号引用)替换为内存指针或者句柄直接引用) 动态链接&#xff1a;程序期间将符号引用替换为直接引用 对象头&#xff1a; 指针压缩&#xff1a; -XX:UseCompressedOops 开启指针压缩 减少内存消耗&#xff1b;大指针在主内存 缓存间移…

6.3 BP神经网络

在多层感知器被引入的同时&#xff0c;也引入了一个新的问题&#xff1a;由于隐藏层的预期输出并没有在训练样例中给出&#xff0c;隐藏层结点的误差无法像单层感知器那样直接计算得到。 为了解决这个问题&#xff0c;反向传播&#xff08;BP&#xff09;算法被引入&#xff0…

centos glibc 升级导致系统崩溃

centos 7.9默认的glibc为2.17&#xff0c;因为要安装一些软件&#xff0c;需要升级到glibc 2.18&#xff0c;而从源码进行编译和安装&#xff0c;安装失败&#xff0c;导致系统崩溃。 系统崩溃首先想到的是利用启动盘进行救援&#xff0c;而利用centos 7.9的启动盘始终无法挂载…

【PHP + 代码审计】数组排序算法

&#x1f36c; 博主介绍&#x1f468;‍&#x1f393; 博主介绍&#xff1a;大家好&#xff0c;我是 hacker-routing &#xff0c;很高兴认识大家~ ✨主攻领域&#xff1a;【渗透领域】【应急响应】 【Java、PHP】 【VulnHub靶场复现】【面试分析】 &#x1f389;点赞➕评论➕收…

银行监管报送系统介绍(五):金融统计数据大集中自动化报送系统——PBOC Report

人民银行金融统计数据大集中自动化报送系统&#xff08;简称PBOC Report&#xff09;&#xff0c;是基于现代计算机网络技术应用基础上&#xff0c;由人行总行设置金融统计数据服务器&#xff0c;建立的一个全国统一的金融统计数据库。 人行针对各银行存贷款、中间业务、网点人…

信号处理--基于FBCSP滤波方法的运动想象分类

目录 理论 工具 方法 代码获取 理论 通用空间模式 (CSP) 算法可以用来有效构建最佳空间滤波器区分&#xff0c;然后实现运动想象的数据中的脑电信号的区分。然而&#xff0c;空间滤波器性能的好坏主要取决于其工作频带。如果脑电信号没有经过滤波或者滤波的频带范围不合适…

FPGA使用XADC测量外部模拟输入电压

一、XADC简介 1.1、特性 Xilinx系列的FPGA中都包含了一个内置的XADC&#xff0c;我们可以通过这个XADC进行一些精度不高的外部模拟信号采样以及FPGA片内传感器信号采集。XADC的分辨率为12位&#xff0c;采样率为1MSPS。 1.2、结构框图 两片XADC&#xff0c;ADC A可用于片内…

SpringJPA 做分页条件查询

前言: 相信小伙伴们的项目很多都用到SpringJPA框架的吧,对于单表的增删改查利用jpa是很方便的,但是对于条件查询并且分页 是不是很多小伙伴不经常写到. 今天我整理了一下在这里分享一下. 话不多说直接上代码: Controller: RestController public class ProductInstanceContr…

Apache HTTP服务器(Linux离线编译安装)

Apache HTTP服务器&#xff08;Linux离线编译安装&#xff09; Apache是普通服务器&#xff0c;本身只支持html即普通网页。可以通过插件支持PHP,还可以与Tomcat连通(单向Apache连接Tomcat,就是说通过Apache可以访问Tomcat资源。反之不然)。 Apache和Tomcat都可以做为独立的w…

8个常见的数据可视化错误以及如何避免它们

在当今以数据驱动为主导的世界里&#xff0c;清晰且具有洞察力的数据可视化至关重要。然而&#xff0c;在创建数据可视化时很容易犯错误&#xff0c;这可能导致对数据的错误解读。本文将探讨一些常见的糟糕数据可视化示例&#xff0c;并提供如何避免这些错误的建议。 本文总结了…

如何使用PHP和RabbitMQ实现消息队列?

前言 今天我们来做个小试验&#xff0c;用PHP和RabbitMQ实现消息队列功能。 前期准备&#xff0c;需要安装好docker、docker-compose的运行环境。 如何使用docker部署php服务_php如何使用docker发布-CSDN博客 一、安装RabbitMQ 1、创建相关目录&#xff0c;执行如下命令。…

计算机网络⑦ —— 网络层协议

1. ARP协议 在传输⼀个 IP 数据报的时候&#xff0c;确定了源 IP 地址和⽬标 IP 地址后&#xff0c;就会通过主机路由表确定 IP 数据包下⼀跳。然⽽&#xff0c;⽹络层的下⼀层是数据链路层&#xff0c;所以我们还要知道下⼀跳的 MAC 地址。由于主机的路由表中可以找到下⼀跳的…

手撕算法-接雨水

描述 分析 i位置能积累的雨水量&#xff0c;等于其左右两边最大高度的最小值。为了能获取i位置左右两边的最大高度。使用动态规划。两个dp数组&#xff1a; leftMaxrightMax 其中 leftMax[i] 代表i位置左边的最大高度rightMax[i] 代表i位置右边的最大高度 初始状态&#x…

Python Flask 自定义过滤器

{{ data.list | li2 }} li2就是自定义的 from flask import Flask, render_templateapp Flask(__name__)app.route("/index") def index():data {name: "张三","age": 18,list: [123123, 41, 123]}return render_template("index2.html…

Redis中RDB中的文件写入

RDB文件的创建与载入。 有两个Redis命令可以用于生成RDB文件&#xff0c;一个是SAVE&#xff0c;另一个是BGSAVE. SAVE命令会阻塞Redis服务器进程&#xff0c;直到RDB文件创建完毕为止&#xff0c;在服务器进程阻塞期间&#xff0c;服务器 不能处理任何命令请求: 127.0.0.1:6…

枚举的详解

枚举的讲解 在C语言中&#xff0c;没有内置的枚举&#xff08;enum&#xff09;数据类型&#xff0c;但我们可以使用整数类型来模拟枚举的行为。C99标准之前&#xff0c;C语言使用#define指令来定义枚举&#xff0c;但这种方式并不安全&#xff0c;因为如果枚举值发生变化&…

【matlab程序】海洋资料的获取与分析--AO/NAO

海洋资料的获取与分析 相关数据代码等资料已上传入群中 海洋资料下载和介绍 AO和NAO指数均取自美国气候预测中心&#xff08;Climate Prediction Center, CPC&#xff09;发布的月平均指数&#xff0c;时间跨度为1950-2022年。由于AO和NAO在冬季最强&#xff0c;因此本文选取…