前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
圈层
工具
发布
首页
学习
活动
专区
圈层
工具
MCP广场
社区首页 >专栏 >redis┃面试官问我redis事务和mysql事务的区别,我。。。。。

redis┃面试官问我redis事务和mysql事务的区别,我。。。。。

作者头像
moon聊技术
发布于 2021-07-28 09:57:38
发布于 2021-07-28 09:57:38
71400
代码可运行
举报
文章被收录于专栏:moon聊技术moon聊技术
运行总次数:0
代码可运行

1

前言

面试官:我看你简历上写了熟悉redis,看来工作中用的很多吧?

我:是的,我们项目中经常用到redis(来,随便问,看我分分钟秒杀你)

面试官:那你给我说说redis的事务和mysql的事务有什么区别吧

我:额。。。事务还有区别????

面试官:比如说redis的事务是不支持原子性和持久性的,包括他们的实现原理等方面也是有很大区别的。

我:学到了。。。。。。

2

正文

事务的四大特性

ACID,指数据库事务正确执行的四个基本要素的缩写。包含:原子性(Atomicity)、一致性(Consistency)、隔离性(Isolation)、持久性(Durability)。

原子性(Atomicity)

说的是一个事物内所有操作就是最小的一个操作单元,要么全部成功,要么全部失败。这是最基本的特性,保证了因为一些其他因素导致数据库异常,或者宕机。

一致性(Consistency)

一个事务可以封装状态改变(除非它是一个只读的)。事务必须始终保持系统处于一致的状态,不管在任何给定的时间并发事务有多少。

一致性有下面特点:

  • 如果一个操作触发辅助操作(级联,触发器),这些也必须成功,否则交易失败。
  • 如果系统是由多个节点组成,一致性规定所有的变化必须传播到所有节点(多主复制)。如果从站节点是异步更新,那么我们打破一致性规则,系统成为“最终一致性”。
  • 一个事务是数据状态的切换,因此,如果事务是并发多个,系统也必须如同串行事务一样操作。

在现实中,事务系统遭遇并发请求时,这种串行化是有成本的, Amdahl法则描述如下:它是描述序列串行执行和并发之间的关系。

“一个程序在并行计算情况下使用多个处理器所能提升的速度是由这个程序中串行执行部分的时间决定的。”

大多数数据库管理系统选择(默认情况下)是放宽一致性,以达到更好的并发性。

隔离性

事物的隔离性,基于原子性和一致性,因为事物是原子化,量子化的,所以,事物可以有多个原子包的形式并发执行,但是,每个事物互不干扰。

但是,由于多个事物可能操作同一个资源,不同的事物为了保证隔离性,会有很多锁方案,当然这是数据库的实现,他们怎么实现的,我们不必深究。

持久性

持久性,当一个事物提交之后,数据库状态永远的发生了改变,即这个事物只要提交了,哪怕提交后宕机,他也确确实实的提交了,不会出现因为刚刚宕机了而让提交不生效,是要事物提交,他就像洗不掉的纹身,永远的固化了,除非你毁了硬盘。

事务命令

mysql:

Begin:显式的开启一个事务

Commit:提交事务,将对数据库进行的所有的修改变成永久性

Rollback:结束用户的事务,并撤销现在正在进行的未提交的修改

redis:

Multi:标记事务的开始

Exec:执行事务的commands队列

Discard:结束事务,并清除commands队列

默认状态

mysql:

mysql会默认开启一个事务,且缺省设置是自动提交,即每成功执行sql,一个事务就会马上commit,所以不能rollback,

redis:

redis默认不会开启事务,即command会立即执行,而不会排队,并不支持rollback

使用方式

mysql(包含两种方式):

用Begin、Rollback、commit显式开启并控制一个 新的 Transaction

执行命令 set autocommit=0,用来禁止当前会话自动commit,控制 默认开启的事务

redis:

用multi、exec、discard,显式开启并控制一个Transaction。

(注意:这里没有强调 “新的” ,因为默认是不会开启事务的)。

实现原理

mysql:

mysql实现事务,是基于undo/redo日志

undo记录修改前状态,rollback基于undo日志实现

redo记录修改后的状态,commit基于redo日志实现

既然是基于redo日志实现记录修改后的状态,那么大家应该也知道,redo日志是innodb专有的,所以innodb会支持事务

在mysql中无论是否开启事务,sql都会被立即执行并返回执行结果,只是事务开启执行后的状态只是记录在redo日志,执行commit之后,数据才会被写入磁盘

(以上内容后面我会详细在mysql篇给大家讲到,大家可以先简单了解下)

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
int insertSelective = serviceOrderMapper.insertSelective(s);

所以,上述代码,insertSelective 将会被立即赋值(无论是否开启事务,只是结果或未被写入磁盘):

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
insertSelective = 受影响的行数;

redis:

redis实现事务,是基于commands队列

如果没有开启事务,command将会被立即执行并返回执行结果,并且直接写入磁盘

如果事务开启,command不会被立即执行,而是排入队列,并返回排队状态(具体依赖于客户端(例如:spring-data-redis)自身实现)。

调用exec才会执行commands队列

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
boolean a = redisTemplate.opsForZSet().add("generalService",orderId,System.currentTimeMillis())boolean a = redisTemplate.opsForZSet().add("generalService",orderId,System.currentTimeMillis())

以上代码如果没有开启事务,操作被立即执行,a将会被立即赋值(true/false)

如果开启事务,操作不会被立即执行,将会返回null值,而a的类型是boolean,所以将会抛出异常:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
java.lang.NullPointerException   空指针异常

Redis事务不支持Rollback(重点)

事实上Redis命令在事务执行时可能会失败,但仍会继续执行剩余命令而不是Rollback(事务回滚)。如果你使用过关系数据库,这种情况可能会让你感到很奇怪。然而针对这种情况具备很好的解释:

  • Redis命令可能会执行失败,仅仅是由于错误的语法被调用(命令排队时检测不出来的错误),或者使用错误的数据类 型操作某个Key: 这意味着,实际上失败的命令都是编程错误造成的,都是开发中能够被检测出来的,生产环境中不应该存在。
  • 由于不必支持Rollback,Redis内部简洁并且更加高效。

redis 事务中的错误

事务期间,可能会遇到两种命令错误:

  • 在调用EXEC命令之前出现错误(COMMAND排队失败)。
  • 例如,命令可能存在语法错误(参数数量错误,错误的命令名称...);
  • 或者可能存在某些关键条件,如内存不足的情况(如果服务器使用maxmemory指令做了内存限制)。

客户端会在EXEC调用之前检测第一种错误。 通过检查排队命令的状态回复(***注意:这里是指排队状态回复,而不是执行结果***),如果命令使用QUEUED进行响应,则它已正确排队,否则Redis将返回错误。如果排队命令时发生错误,大多数客户端将中止该事务并清除命令队列。然而:

  • Redis 2.6.5之前,这种情况下,在EXEC命令调用后,客户端会执行命令的子集(成功排队的命令)而忽略之前的错误。
  • Redis 2.6.5开始,服务端会记住在累积命令期间发生的错误,当EXEC命令调用时,将拒绝执行事务,并返回这些错误,同时自动清除命令队列
  • 示例如下:
代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
>MULTI
+OK
>INCR a b c
-ERR wrong number of arguments for 'incr' command

这是由于INCR命令的语法错误,将在调用EXEC之前被检测出来,并终止事务(version2.6.5+)。

  • 在调用EXEC命令之后出现错误。
  • 例如,使用错误的值对某个key执行操作(如针对String值调用List操作)

EXEC命令执行之后发生的错误并不会被特殊对待即使事务中的某些命令执行失败,其他命令仍会被正常执行

  • 示例如下:
代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
>MULTI
+OK
>SET a 3
+QUEUED
>LPOP a
+QUEUED
>EXEC
*2
+OK
-ERR Operation against a key holding the wrong kind of value>
  • EXEC返回一个包含两个元素的字符串数组,一个元素是OK,另一个是-ERR……
  • 能否将错误合理的反馈给用户这取决于客户端library(如:Spring-data-redis.redisTemplate)的自身实现。
  • 需要注意的是,即使命令失败,队列中的所有其他命令也会被处理----Redis不会停止命令的处理。

3

结语

今天moon给大家讲了redis的事务和mysql 的事务到底有什么区别,看完这篇文章大家也应该大致明白了

redis和mysql 的事务在默认状态,使用方式,实现原理等方面都是有很大区别的

mysql的事务完美的支持了事务的四大特性,而redis事务只保证了其中的一致性和隔离性,不满足原子性和持久性

当面试官问到你的时候想必你也能够对答如流了吧。

redis系列的文章已经更新的差不多了,下一篇我们来聊聊在如何使用redis做分布式锁。

记得点赞

本文参与 腾讯云自媒体同步曝光计划,分享自微信公众号。
原始发表:2020-10-09,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 moon聊技术 微信公众号,前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体同步曝光计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
暂无评论
推荐阅读
编辑精选文章
换一批
Redis09-Redis事务
上一篇我们介绍了Redis的持久化[1],这一篇我们接着来学习Redis的事务。将从如下几个方面进行阐述,事务的介绍,Redis事务的介绍,Redis事务与数据库事务的区别。
码农飞哥
2021/08/18
2660
腾讯二面:Redis 事务支持 ACID 么?
「内心独白:小意思,不就 ACID 嘛,转眼一想,我面试的可是技术专家,不会这么简单的问题吧」
码哥字节
2021/10/21
6510
【玩转Redis面试第2讲】面试官再问Redis事务把这篇文章扔给他
原子性:一个事务(transaction)中的所有操作,要么全部完成,要么全部不完成,不会结束在中间某个环节。事务在执行过程中发生错误,会被恢复(Rollback)到事务开始前的状态,就像这个事务从来没有执行过一样。
爱笑的架构师
2020/09/24
7670
【玩转Redis面试第2讲】面试官再问Redis事务把这篇文章扔给他
Redis面试(七):事务
使用起来也很简单,可以用 multi 开启一个事务,然后将多个命令入队到事务的队列中,最后由exec命令触发事务,执行事务中的所有命令。
传说之下的花儿
2023/09/26
2800
Redis面试(七):事务
Redis的事务机制
用于标记事务块的开启。MULTI执行之后,Redis会将后续的命令逐个放到一个缓存队列中,当EXEC命令被调用时,所有队列中的命令才会被原子化执行。
全栈程序员站长
2022/06/29
4820
Redis事务处理机制分析与总结
事务是一组具备原子性操作的命令集合。在这一组命令中,要么全部执行成功,要么全部执行失败。
兔云小新LM
2021/01/28
8370
Redis事务
①原子性:redis原子性是指将多个操作打包在一起,要么全都执行,要么全都不执行。注意:这里跟MySQL事务中的原子性相比,redis原子性不管这些操作有没有成功,它不管!如果事务中有些操作失败了,redis说失败就失败吧。而MySQL则不行,一旦有操作失败,则全部回滚!(有部分观点任务,redis没有原子性,因为以MySQL事务的原子性作为标杆,原子性必须要么执行成功,要么不执行)
二肥是只大懒蓝猫
2024/07/23
1090
Redis事务
Redis 事务的本质是一组命令的集合。事务支持一次执行多个命令,一个事务中所有命令都会被序列化。在事务执行过程,会按照顺序串行化执行队列中的命令,其他客户端提交的命令请求不会插入到事务执行命令序列中。总结说:redis事务就是一次性、顺序性、排他性的执行一个队列中的一系列命令。
用户3876103
2024/08/28
1680
【Redis入门到精通八】Redis事务与MySQL事务对比
MySQL基于事务的隔离性,还延伸出来了三种事务的隔离级别,分别是脏读,不可重复读,幻读从上到下隔离级别越来越高,相应的性能开销也越来越大。
小皮侠
2024/10/01
1710
【Redis入门到精通八】Redis事务与MySQL事务对比
不支持原子性的 Redis 事务也叫事务吗?
为了确保连续多个操作的原子性,我们常用的数据库都会有事务的支持,Redis 也不例外。但它又和关系型数据库不太一样。
海星
2020/09/07
2.5K0
Redis事务概述:解开命令执行的完整故事
reactor网络模型,其组成是IO多路复用+非阻塞IO;IO职责是IO检测和IO操作;事件是异步事件处理流程,先注册事件,然后在事件循环中处理事件,注册事件的时候提供回调函数。
Lion 莱恩呀
2024/11/06
800
Redis事务概述:解开命令执行的完整故事
Redis事务
ACID (原子性(atomicity,或称不可分割性)、 一致性(consistency)、 隔离性(isolation,又称独立性)、 持久性(durability) )
暴躁的程序猿
2022/05/10
2280
redis | 十二、redis之事务
Redis 的事务只是一组命令的集合,一个事务中的所有命令都会被序列化,执行过程中按照顺序执行,并且其它会话提交的命令不会插入到事务执行的命令序列中
雨中散步撒哈拉
2022/09/21
3280
Redis事务
和其它数据库一样,Redis作为NoSQL数据库也同样提供了事务机制。在Redis中,MULIT,EXEC,DISCARD,WATCH这个四个命令是实现事务的基石,Redis中事务的实现特征
莫问今朝
2018/08/31
7380
Redis事务
redis事务
本文记录一些redis事务相关的原理。 1、基本概念 1)什么是redis的事务? 简单理解,可以认为redis事务是一些列redis命令的集合,并且有如下两个特点: a)事务是一个单独的隔离操作:事务中的所有命令都会序列化、按顺序地执行。事务在执行的过程中,不会被其他客户端发送来的命令请求所打断。 b)事务是一个原子操作:事务中的命令要么全部被执行,要么全部都不执行。 2)事务的性质ACID 一般来说,事务有四个性质称为ACID,分别是原子性,一致性,隔离性和持久性。 a)原子性atomicity:red
用户1225216
2018/03/05
6370
Redis系列之事务机制
学习mysql数据库的时候,我们知道了事务的ACID特性,Redis也是支持事务的,不过和数据库的事务又有什么区别?在mysql数据库中,我们使用begin开启事务,提交是commit,回滚是rollback,然后Redis中的事务是怎么一回事?redis的事务其实可以看做是一组命令按照顺序,串行执行队列中的命令,其它客户端的命令不会写入到这个队列中。总的来说,redis事务就是一次性、顺序性、排他性的执行一个队列中的一组命令
SmileNicky
2023/12/18
1760
Redis篇:事务和lua脚本的使用
除了 MULTI、WATCH、EXEC 命令,还有其他的方式可做到 redis 原子性和隔离性吗?有的,lua 脚本;redis 内置了lua的执行环境,并自带了一些 lua 函数库。redis 执行 lua 时,会启动一个伪客户端去执行脚本里的 redis 命令
潜行前行
2021/12/02
2.3K0
从Redis事务到Redis pipeline
相信对关系性数据库有使用经验的,都对事务操作很熟悉,为了确保连续多个操作的原子性,我们常用的数据库都会有事务的支持,Redis 也不例外;但它又和关系型数据库支持的事务不太一样
Kevinello
2022/08/19
8540
从Redis事务到Redis pipeline
Redis系统学习之事物
Redis事物操作 Redis 事务可以一次执行多个命令, 并且带有以下三个重要的保证: 批量操作在发送 EXEC 命令前被放入队列缓存。 收到 EXEC 命令后进入事务执行,事务中任意命令执行失败,其余的命令依然被执行。 在事务执行过程,其他客户端提交的命令请求不会插入到事务执行命令序列中。 一个事务从开始到执行会经历以下三个阶段: 开始事物->命令入队->执行事务|取消事物 单个 Redis 命令的执行是原子性的,但 Redis 没有在事务上增加任何维持原子性的机制,所以 Redis 事务的执行并不是原
彼岸舞
2021/05/07
4030
Redis系统学习之事物
Redis事务
Redis 事务的基本功能由 MULTI、EXEC、DISCARD 及 WATCH 等命令实现。其中,
用户7353950
2022/05/10
4300
Redis事务
相关推荐
Redis09-Redis事务
更多 >
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档
本文部分代码块支持一键运行,欢迎体验
本文部分代码块支持一键运行,欢迎体验