【Redis7】--4.事务、管道、发布和订阅

2023-09-18 21:35:32

事务

1.Redis事务

Redis事务可以一次执行多个命令,本质是一组命令的集合。一个事务中的所有命令都会序列化,按顺序地串行化执行而不会被其他命令插入,不许加塞

在一个队列中,一次性、顺序性、排他性的执行一系列命令

Redis 事务使用 MULTI、EXEC、WATCH、DISCARD 和 UNWATCH 这些命令来实现。

2.Redis事务特性

Redis事务vs数据库事务

image-20230918202409407

  • 单独的隔离操作:事务中的所有命令都会序列化、按顺序地执行。事务在执行的过程中,不会被其他客户端发送来的命令请求所打断
  • 不保证原子性:Redis的事务不保证原子性,也就是不保证所有指令同时成功或同时失败,只有决定是否开始执行全部指令的能力,没有执行到一半进行回滚的能力
  • 排他性:Redis会保证一个事务内的命令依次执行,不会被其他命令插入

3.Redis事务命令

MULTI:开启事务

EXEC:执行事务

DISCARD:取消事务

WATCH key [key ...]:监控指定的key

UNWATCH :取消监控。

3.1MULTI

标记一个事务块的开始。执行的命令都会放到一个队列中,通过EXEC命令统一执行。

3.2EXEC

执行事务队列中的命令。

image-20230918204251779

3.3DISCARD

放弃事务。在开启了事务后,若不想执行命令了,可以通过DISCARD命令来取消事务。

image-20230918204340697

3.4WATCH

监控指定的key的变化,要先开启监控,再开启事务。若监控的数据被篡改了,则事务中无法再对其修改,会返回nil。

执行完EXEC命令后,之前加的监控都会失效。

Redis使用Watch来提供乐观锁定,类似于CAS

回顾一下悲观锁、乐观锁:

悲观锁(Pessimistic Lock), 顾名思义,就是很悲观,每次去拿数据的时候都认为别人会修改,所以每次在拿数据的时候都会上锁,这样别人想拿这个数据就会block直到它拿到锁。

乐观锁(Optimistic Lock), 顾名思义,就是很乐观,每次去拿数据的时候都认为别人不会修改,所以不会上锁,但是在更新的时候会判断一下在此期间别人有没有去更新这个数据。

乐观锁策略:提交版本必须 大于 记录当前版本才能执行更新

初始化k1和balance两个key,先监控再开启multi,保证两个key变动在同一个事务内

image-20230918204712152

有加塞篡改:

image-20230918204846786

3.5UNWATCH

取消监控。在对某个key监控后,已经发现了它被篡改过了,可以使用unwatch命令取消对该key的监控。

image-20230918204927500

4.不保证原子性

4.1"全体连坐"

要么都执行,要么都不执行。

image-20230918205032972

4.2"冤头债主"

错误的命令报错归报错,正确的命令依旧执行,即使在错误命令之后也会执行。

image-20230918205131077

注意和传统数据库事务区别,不一定要么一起成功要么一起失败

5.事务执行流程

(1)开启:以MULTI命令开启一个事务

(2)入队:将多个命令假如到事务队列中,接到这些命令并不会立即执行。

(3)执行:由EXEC命令执行事务队列中的命令。

管道

image-20230918210238572

image-20230918210245630

image-20230918210307655

Redis 管道(Pipeline)是一种在客户端和 Redis 服务器之间建立的双向通道,它可以让客户端在一次请求中发送多个命令并一次性接收多个命令的响应结果。通过使用 Redis 管道,客户端可以减少网络通信的次数,从而提高 Redis 的吞吐量和性能。

在传统的 Redis 操作中,每个命令都需要通过网络发送到 Redis 服务器,然后等待 Redis 服务器返回响应结果后再进行下一个命令的操作,这样就会产生大量的网络通信开销。而使用 Redis 管道,客户端可以将多个命令一次性发送到 Redis 服务器,然后一次性接收所有命令的响应结果,从而减少网络通信的次数和开销。

批处理命令变种优化措施,类似Redis的原生批命令(mget和mset)

1.pipeline的使用

首先创建一个文件,写入需要执行的命令集。
在Linux终端使用cat cmd.txt | redis-cli -a 123456 --pipe命令将命令集传输到服务器。
(管道符”|“表示将前面命令的结果集作为参数传输给后面的命令)

image-20230918210355174

2.pipeline小总结

  • Pipeline与原生批量命令对比:

    • 原生批量命令(例如mset、mget)具有原子性,pipeline是非原子性。
    • 原生批量命令一次只能执行一种命令,pipeline支持批量执行不同命令。
    • 原生批命令是redis服务端实现,而pipeline需要redis服务端和客户端共同完成。
  • Pipeline与事务对比:

    • 事务具有原子性,管道不具有原子性。
    • 管道一次性将命令发送给服务器,事务是一条一条的发,事务只有在接收到EXEC命令后才会执行。
    • 执行事务时会阻塞其他命令的执行,而执行管道中的命令不会。
  • 使用Pipeline注意事项:

    • pipeline缓冲的指令只是会依次执行,不保证原子性,如果执行中指令发生异常,还会继续执行后续的指令。
    • 使用pipeline传输的命令也不能太多,如果数据量大客户端的阻塞时间可能会过久,同时服务端此时也被迫回复一个队列答复,占用很多内存。

发布和订阅

Redis发布和订阅(Publish/Subscribe,简称 Pub/Sub)是一种消息传递模式,用于在Redis中实现消息的发布和订阅,发送者发送消息,订阅者接收消息,可以实现进程之间的消息传递

在 Redis 中,发布者(Publisher)可以将消息发送到一个或多个频道(Channel),订阅者(Subscriber)可以订阅一个或多个频道,以接收发布者发送的消息。当发布者在某个频道上发布一条消息时,所有订阅该频道的订阅者都会收到这条消息。

Redis Pub/Sub 是基于消息传递的异步通信模型,可以用于构建实时系统、聊天室、实时广播等应用场景。

image-20230918210940173

1.常用命令

1.1SUBSCRIBE

SUBSCRIBE channel [channel ...]:订阅一个或多个频道

一旦客户进入了订阅状态,客户端就只能接受订阅相关的命令SUBSCRIBE、PSUBSCRIBE、UNSUBSCRIBE和PUNSUBSCRIBE,除了这些命令,其他命令一律失效。按Ctrl+C结束订阅状态。

返回值:发布类型、频道名称、第几个频道

image-20230918212312940

1.2PUBLISH

PUBLISH channel message:发布消息到指定频道。

返回值为收到消息的客户端数量。

1.3PSUBSCRIBE

PSUBSCRIBE pattern [pattern ...]:按照匹配模式批量订阅。

支持的模式有:?表示任意一个字符;*表示任意数量的任意字符;[]表示中括号中的指定字符。比如:
h?llo:可以匹配hallo、hbllo、hello…
h*llo:可以匹配hello、heeello、habcdello…
h[abc]llo:只能匹配hallo、hbllo、hcllo

1.4PUBSUB

PUBSUB 是自省命令,能够检测PUB/SUB子系统的状态。

PUBSUB CHANNELS [pattern] :返回当前活跃的频道。

只会统计使用SUBSCRIBE订阅的频道。

PUBSUB NUMSUB channel [channel ...]:返回指定频道订阅者的个数。

只会统计使用SUBSCRIBE订阅的订阅者个数。

PUBSUB UNMPAT:返回订阅模式(PSUBSCRIBE)的数量。

这个命令返回的不是订阅模式的订阅者数量, 而是所有Redis客户端(订阅者)订阅的所有模式的数量总和。

image-20230918212434855

image-20230918212440587

1.5UNSUBSCRIBE

UNSUBSCRIBE channel [channel ...]:指示客户端退订指定频道,若没有指定频道则退订所有频道。

image-20230918212649890

1.6 PUNSUBSCRIBE

PUNSUBSCRIBE pattern [pattern ...]:指示客户端退订指定模式,若没有提供模式则退定所有模式。

2.总结

Redis可以实现消息中间件MQ的功能,通过发布订阅实现消息的引导和分流,但是专业的事情交给专业的中间件处理,redis主要做好分布式缓存功能

  • 发布的消息在Redis系统不能持久化,因此必须先执行订阅,再等待消息发布,如果先发布了消息且该消息没有订阅者接收,那么该消息被直接丢弃。
  • 消息只管发送,对于发布者而言消息是即发即失的,也没有ACK机制,无法保证消息是否消费成功。
  • Redis5.0新增了Stream数据结构,不但支持多播,还支持数据持久化,比Pub/Sub更加强大。
更多推荐

android系统目录结构

文章目录android系统目录结构问答偏好设置保存在哪里在应用设置中点击清除数据,清除的是什么在应用设置中点击清除缓存,清除的是什么参考android系统目录结构/-system(一般只有root权限才能访问)-data-app(存放应用程序的APK文件)-data(内部存储)-<安装的应用包名>-app_textur

lock和synchronized的区别

lock和synchronized都是在多线程环境下用于保护共享资源的机制,但它们有一些重要的区别:实现方式:synchronized是Java语言内置的关键字,可以用于方法或代码块级别的同步。Lock是一个接口,位于java.util.concurrent.locks包下,提供了更灵活的锁定机制。灵活性:Lock提供

SQL-4大板块(存储过程、函数、视图、触发器)

一、存储过程(做复杂运算)1.做复杂运算,是对变量做运算;2.可以对多个表进行update、insert、delete、select、query;3.可以在最终结果返回多个表,但是对对接环境有苛刻要求,比如:VB不支持接收返回多个表4.当一个查询语句无法实现,或者语句查询速度很慢时,想提高效率就会用到存储过程。先把需要

MongoDB——将时间戳转换为日期

在MongoDB中将时间戳转换为日期从timestamp转换为日期取决于我们保存时间戳的类型。它是对象、数字还是字符串类型?我们可以在mongoshell上使用以下命令检查字段的类型。在本教程中,我们将学习如何将时间戳转换为数字、字符串或对象类型的日期。检查字段类型://MongoDB5.0.8>typeofdb.co

滑动时间窗口的思想和实现,环形数组,golang

固定时间窗口在开发限流组件的时候,我们需要统计一个时间区间内的请求数,比如以分钟为单位。所谓固定时间窗口,就是根据时间函数得到当前请求落在哪个分钟之内,我们在统计的时候只关注当前分钟之内的数量,即[0s,60s],因为流量并不是均匀的,所以就会出现,在两个分钟之间超过阈值,1分50秒时来了150个请求,在2分10秒时来

图像语义分割概述

图像语义分割概述一、图像语义分割概念图像语义分割(ImageSemanticSegmentation)是一项计算机视觉任务,其目标是将输入的图像分割成多个区域,并为每个像素分配一个语义类别标签,以表示该像素属于图像中的哪个物体或区域。与其他图像分割任务不同,图像语义分割不仅关注于分割图像,还要理解图像中不同部分的语义含

【C++】面向对象编程示例 ( 案例需求 | Visual Studio 创建类 | 类的声明 | 类的实现 | 类的调用 )

文章目录一、案例需求二、VisualStudio创建类三、类的声明四、类的实现五、类的调用一、案例需求使用C++面向对象,抽象出一个立方体类;立方体有长/宽/高/面积/体积私有成员变量,以及访问这些成员变量的公共成员方法;还提供立方体的对比函数,对比2个立方体对象是否相等;二、VisualStudio创建类在Visua

【C++】类的声明 与 类的实现 分开 ① ( 类的声明 与 类的实现 常用用法 | Visual Studio 2019 中创建类的头文件和源文件 | 确保头文件包含一次 )

文章目录一、类的声明与类的实现分开1、类的声明与类的实现常用用法2、VisualStudio2019中创建类的头文件和源文件3、Student.h类头文件解析4、确保头文件包含一次一、类的声明与类的实现分开1、类的声明与类的实现常用用法在之前的博客中,定义的class类,定义类时同时也完成了实现;但是在C++语言实际开

“源启2.0”:从自上而下的解构,到自下而上的重构

从垂直打穿、到应用重构,中电金信赋能行业数字化转型之路既“向下走”、也“向上看”。“向上”先理解和吃透客户的企业战略,进而自上而下地将企业战略拆解为业务架构,“向下”再将业务架构拆解为应用架构和数据架构,并进一步对齐技术架构。而在此过程中,上至“应用重构”,下至“数字基础设施重构”就都已不是问题。这就是“源启2.0”的

【校招VIP】测试方案分析之压力测试

考点介绍:软件压力测试是校招面试里面经常会碰到的题型。基本思路是在计算机数量较少或系统资源匮乏的条件下运行测试。要求面试者了解压力测试的基本概念,压测的目的,压测的要求以及说出实例。测试方案分析之压力测试-相关题目及解析内容可点击文章末尾链接查看!一、考点试题1.软件质量管理(QM)应有质量保证(QA)和质量控制(QC

以神龙出行小程序为例,说一些网站技术

注册和登录功能:用户注册和登录可以使用手机号验证、第三方登录等方式来实现。这需要与后台服务器进行数据交互,并进行身份验证。数据存储和管理:用户的个人信息和常用地址需要进行存储和管理。这可以通过数据库来实现,如关系型数据库或NoSQL数据库。智能匹配和推荐:搬家、拉货和代驾服务需要智能匹配合适的车型和司机。这可以利用算法

热文推荐