IT数码 购物 网址 头条 软件 日历 阅读 图书馆
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
图片批量下载器
↓批量下载图片,美女图库↓
图片自动播放器
↓图片自动播放器↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁
 
   -> 大数据 -> MySQL理论:脏读、不可重复读、幻读 -> 正文阅读

[大数据]MySQL理论:脏读、不可重复读、幻读

1. 脏读(dirty read)

脏读是指事务读取到其他事务未提交的数据

例如:有事务A、B和一条记录:id为1,name为张三

B首先进行更新操作,将name的值由张三改为张老三,但还未提交事务

begin;
update stu set name = '张老三' where id = 1

然后A进行查询操作,查询姓名为张老三

select name from stu where id = 1

然后B由于某种原因被回滚,name自然从张老三恢复到了张三

A就会出现一个问题,读取的是张老三,但数据库中实际存储的是张三,我到底是用哪一个呢?

这就是脏读,事务A读取到事务B未提交的数据

脏读是我们在整个数据库操作中最普遍的一个现象,但是在日常开发中我们几乎不会遇到脏读,原因以后说

看图更容易理解
在这里插入图片描述

2. 不可重复读(non-repeatable read)

不可重复读是指在同一次事务中前后查询不一致的问题

例如:

A先查询了一条记录,name为张三

select name from stu where id = 1

然后B执行了更新,并且提交

begin;
update stu set name = '张老三' where id = 1;
commit

A再次按相同条件查询该记录,name为张老三

select name from stu where id = 1

A就会出现一个问题,同样的查询语句,两次的执行结果却不一致(不重复)

第二次查询得到的name应该还是预期的张三,因为很明确,A并没有对name进行修改,但是在并发环境下,假如两条select语句间有另外一个事务对name执行了update操作并提交了,把张三的名字改为了张老三,A再次执行同样查询导致name由张三变为张老三

A就有意见了,我明明没有对name进行修改,可是为什么后来变成张老三了?谁改我数据了?

这就是不可重复读:同一次事务中前后查询不一致

它会让我们的程序运行变得不可预期、不可控

看图更容易理解

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-tJ2rTdRb-1631029938002)(C:\Users\PC\AppData\Roaming\Typora\typora-user-images\image-20210907231354003.png)]

3. 幻读(phantom read)

幻读是一次事务中前后数据量发生变化,用户产生不可预料的问题

分为并发删除和并发插入的情况

1 并发插入

A先查询了一条记录,name为张三

select name from stu where id = 1

B插入了一条数据,name为李四

insert into stu(name) values('李四')

当A再次按相同条件查询

select name from stu where id = 1

结果多了一条李四的记录,刚才明明只有一条记录,怎么多出来一条?好像出现了幻觉

2 并发删除

将上面的插入操作换成删除操作,B删除了张三的记录

delete from stu where id = 1

结果A会发现张三的记录神秘的消失了,明明刚才还有的,怎么又没了,再次出现了幻觉

这就是幻读,看图更容易理解

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-ma99aVHO-1631029938005)(C:\Users\PC\AppData\Roaming\Typora\typora-user-images\image-20210907234039087.png)]

4. 不可重复读和幻读的比较

1 幻读和不可重复读都是读取了另一条已经提交的事务(这点就脏读不同),所不同的是不可重复读查询的都是同一个数据项,而幻读针对的是一批数据整体(比如数据的个数)。
2 不可重复读的重点是修改,幻读的重点在于插入或者删除
3 但如果从控制的角度来看, 两者的区别就比较大
对于前者, 只需要锁住满足条件的记录
对于后者, 要锁住满足条件及其相近的记录

5. 总结

脏读是指事务读取到其他事务正在处理的未提交数据

不可重复读指在并发更新时,另一个事务前后执行相同条件的查询得到的数据不一致

幻读指并发插入、删除时,另一个事务前后执行相同条件的查询得到的数据不一致

6. 解决方法

如何解决以上问题,事务的隔离级别就派上用场了

禁止写时读,避免了“脏读”,对应隔离级别read committed。

禁止读时写,避免了“不可重复读”,对应隔离级别repeatable read。

而为了避免“幻读”,干脆把整个表给锁住了,只能是serialize了。

隔离级别越高,并行度越低,付出的代价越大。

  大数据 最新文章
实现Kafka至少消费一次
亚马逊云科技:还在苦于ETL?Zero ETL的时代
初探MapReduce
【SpringBoot框架篇】32.基于注解+redis实现
Elasticsearch:如何减少 Elasticsearch 集
Go redis操作
Redis面试题
专题五 Redis高并发场景
基于GBase8s和Calcite的多数据源查询
Redis——底层数据结构原理
上一篇文章      下一篇文章      查看所有文章
加:2021-09-08 10:48:29  更:2021-09-08 10:49:47 
 
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁

360图书馆 购物 三丰科技 阅读网 日历 万年历 2025年1日历 -2025/1/18 14:33:39-

图片自动播放器
↓图片自动播放器↓
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
图片批量下载器
↓批量下载图片,美女图库↓
  网站联系: qq:121756557 email:121756557@qq.com  IT数码