当前位置: 首页 > article >正文

【Redis经典面试题七】Redis的事务机制是怎样的?

目录

一、Redis的事务机制

二、什么是Redis的Pipeline?和事务有什么区别?

三、Redis的事务和Lua之间有哪些区别?

3.1 原子性保证

3.2 交互次数

3.3 前后依赖

3.4 流程编排

四、为什么Lua脚本可以保证原子性?

五、为什么Redis不支持回滚?


一、Redis的事务机制

Redis中是支持事务的,他的事务主要目的是保证多个命令执行的原子性,即要在一个原子操作中执行,不会被打断。

需要注意的是,Redis的事务是不支持回滚的。从 Redis 2.6.5 开始,服务器将会在累积命令的过程中检测到错误。然后,在执行 EXEC 期间会拒绝执行事务,并返回一个错误,同时丢弃该事务。如果事务执行过程中发生错误,Redis会继续执行剩余的命令而不是回滚整个事务。

也就是说,如果一套事务的命令有语法错误,比如你写个压根没有的命令,那Redis会帮你检测到,并且整个事务的命令都不会执行。但要是执行过程中出错,比如对list进行incr操作,那还是会执行下面的命令,并且不会回滚。

redis其实挺尴尬的,像Mysql事务只要其中有一个命令出错,其它全不执行。但redis却不是这样,提供事务功能给你就不错了,凑合用吧,你程序员多认真点,尽量不要自己弄错就没事。如果真弄错了,那你程序员就自己手动回滚吧。

Redis事务相关的命令主要有以下几个:

  • MULTI:标记一个事务块的开始。
  • DISCARD:取消事务,放弃执行事务块内的所有命令。
  • EXEC:执行所有事务块内的命令。
  • UNWATCH:取消 WATCH 命令对所有 key 的监视。
  • WATCH key [key ...]:监视一个(或多个)key,如果在事务执行之前这个(或这些)key 被其他命令所改动,那么事务将被打断。

二、什么是Redis的Pipeline?和事务有什么区别?

Redis 的 Pipeline 机制是一种用于优化网络延迟的技术,主要用于在单个请求/响应周期内执行多个命令。在没有 Pipeline 的情况下,每执行一个 Redis 命令,客户端都需要等待服务器响应之后才能发送下一个命令。这种往返通信尤其在网络延迟较高的环境中会显著影响性能。

在 Pipeline 模式下,客户端可以一次性发送多个命令到 Redis 服务器,而无需等待每个命令的响应。Redis 服务器接收到这批命令后,会依次执行它们并返回响应。

所以,Pipeline通过减少客户端与服务器之间的往返通信次数,可以显著提高性能,特别是在执行大量命令的场景中。

但是,需要注意的是,Pipeline是不保证原子性的。他的多个命令都是独立执行的,Redis并不保证这些命令可以以不可分割的原子操作进行执行。这是Pipeline和Redis的事务的最大的区别。

虽然都是执行一些相关命令,但是Redis的事务提供了原子性保障,保证命令执行以不可分割、不可中断的原子性操作进行,而Pipeline则没有原子性保证。

但是他们在命令执行上有一个相同点,那就是如果执行多个命令过程中,有一个命令失败了,其他命令还是会被执行,而不会回滚的。

三、Redis的事务和Lua之间有哪些区别?

Redis中,事务和Lua都是保证原子性的手段。当我们有多个命令要执行,并希望它们以原子性方式执行的时候,就会考虑使用事务或者Lua脚本。那么他们之间有哪些区别呢?

3.1 原子性保证

事务和Lua都是可以保证原子性操作的,但是,这里说的原子性我们提过很多次,指的是不可拆分、不可中断的原子性操作。所以,需要注意的是,不管是Redis的事务还是Lua,都没办法回滚,一旦执行过程中有命令失败了,都是不支持回滚的。

但是,Redis的事务在执行过程中,如果有某一个命令失败了,是不影响后续命令的执行的,而Lua脚本中,如果执行过程中某个命令执行失败了,是会影响后续命令执行的。

3.2 交互次数

在Redis的事务执行时,每一条命令都需要和Redis服务器进行一次交互。我们可以在Redis事务过程中,在MULTI和EXEC之间发送多个Redis命令给Redis服务器,这些命令会被服务器缓存起来,但并不会立即执行。但是每一条命令的提交都需要进行一次网络交互。

而Lua脚本则不需要,只需要一次性地把整个脚本提交给Redis即可。网络交互比事务要少。

3.3 前后依赖

在Redis的事务中,事务内的命令都是独立执行的,并且在没有执行EXEC命令之前,命令是没有被真正执行的,所以后续命令是不会也不能依赖于前一个命令的结果的。

而在Lua脚本中是可以依赖前一个命令的结果的。Lua脚本中的多个命令是依次执行的,我们可以利用前一个命令的结果进行后续的处理。

3.4 流程编排

借助Lua脚本,我们可以实现非常丰富的各种分支流程控制,以及各种运算相关操作。而Redis的事务本身是不支持这些操作的。

四、为什么Lua脚本可以保证原子性?

原子性在并发编程中和在数据库中是两种不同的概念。

在数据库中,事务的ACID中原子性指的是“要么都执行要么都回滚”。在并发编程中,原子性指的是“操作不可拆分、不被中断”。

Redis既是一个数据库,又是一个支持并发编程的系统,所以它的原子性有两种。因此,我们需要明确清楚,在问“Lua脚本保证Redis原子性”的时候,指的到底是哪个原子性。

Lua脚本可以保证原子性,因为Redis会将Lua脚本封装成一个单独的事务,而这个单独的事务会在Redis客户端运行时,由Redis服务器自行处理并完成整个事务。如果在这个进程中有其他客户端请求的时候,Redis将会把它暂存起来,等到Lua脚本处理完毕后,才会再把被暂存的请求恢复。

这样就可以保证整个脚本是作为一个整体执行的,中间不会被其他命令插入。但是,如果命令执行过程中命令产生错误,事务是不会回滚的,将会影响后续命令的执行。

也就是说,Redis保证以原子方式执行Lua脚本,但是不保证脚本中所有操作要么都执行或者都回滚。

那就意味着,Redis中Lua脚本的执行,可以保证并发编程中不可拆分、不被中断的这个原子性,但是没有保证数据库ACID中要么都执行要么都回滚的这个原子性。

五、为什么Redis不支持回滚?

我们都知道,Redis是不支持回滚的,即使是Redis的事务和Lua脚本,在执行的过程中,如果出现了错误,也是无法回滚的。可是,为什么呢?

在Redis的官网文档中明确提到过,不支持回滚: Transactions | Docs

不支持回滚主要的原因是支持回滚将对 Redis 的简洁性和性能产生重大影响。

总结一下,因为Redis的设计就是简单、高效等,所以引入事务的回滚机制会让系统更加复杂,并且影响性能。从使用场景上来说,Redis一般都是被用作缓存的,不太需要很复杂的事务支持。当人们需要复杂的事务时会考虑持久化的关系型数据库。相比于关系型数据库,Redis是通过单线程执行的,在执行过程中,出现错误的概率比较低,并且这些问题一般在编译阶段都应该被发现,所以就不太需要引入回滚机制。


http://www.kler.cn/a/469540.html

相关文章:

  • 设计模式——泛型单例类
  • 如何删除 Docker 中的悬虚镜像?
  • C语言的正则表达式
  • 【HarmonyOS】鸿蒙应用如何进行页面横竖屏切换以及注意事项,自动切换横竖屏,监听横竖屏
  • C++进阶——用Hash封装unordered_map和unordered_set
  • Uniapp Android 本地离线打包(详细流程)
  • CISA 列入高风险漏洞 CVE-2024-12356 至已被利用漏洞名单
  • 简述Linux的信号处理
  • 深入Android架构(从线程到AIDL)_11 线程之间的通信架构
  • 代码随想录 day62 第十一章 图论part11
  • windows C#-字符串和字符串字面量(三)
  • MySQL数据库基础 === DDL DML
  • AlphaPi相关硬件驱动提取
  • 软件工程期末大复习(六)面向对象分析
  • Spring boot 项目 Spring 注入 代理 并支持 代理对象使用 @Autowired 去调用其他服务
  • javascript 绘制图表的几种方式
  • 解决iNodeClient客户端出现查询SSL VPN网关参数失败的问题
  • springboot573学院个人信息管理系统(论文+源码)_kaic
  • WebRtc01:课程导学、框架介绍
  • 为LLM加速:博查搜索API助力下一代AI应用开发
  • 【优选算法】查找总价格为目标值的两个商品
  • Problem - Contest
  • 在Vue项目中使用Element UI实现一个树形功能,可以通过el-tree组件来实现,树的节点内容从配置文件中读取。
  • HarmonyOS开发:关于帧动画使用分享
  • 群晖上安装Tomcat运行环境
  • 内蒙古水系详细很全shp格式arcgis软件无偏移坐标下载后内容测评