昨天接到阿里的电话面试,对方问了一个在MySQL当中,什么是幻读。当时一脸懵逼,凭着印象和对方胡扯了几句。面试结束后,赶紧去查资料,才发现之前对幻读的理解完全错误。下面,我们就聊聊幻读。

要说幻读,就要从MySQL的隔离级别说起。MySQL的4钟隔离级别分别是:

Read Uncommitted(读取未提交内容)

在该隔离级别,所有事务都可以看到其他未提交事务的执行结果。本隔离级别很少用于实际应用,因为它的性能也不比其他级别好多少。读取未提交的数据,也被称之为脏读(Dirty Read)。

脏读的具体示例如下:

时间点 事务A 事务B
1 开启事务
2 开启事务
3 查询数据为100条
4 insert一条数据
5 再查询,结果为101条

在时间点5,事务A再次查询数据时,事务B并没有提交事务,但是,新的数据也被事务A查出来了。这就是脏读。

Read Committed(读取提交内容)

这是大多数数据库系统的默认隔离级别(但不是MySQL默认的)。它满足了隔离的简单定义:一个事务只能看见已经提交事务所做的改变。这种隔离级别 也支持所谓的不可重复读(Nonrepeatable Read),因为同一事务的其他实例在该实例处理其间可能会有新的commit,所以同一select可能返回不同结果。

时间点 事务A 事务B
1 开启事务
2 开启事务
3 查询数据为100条
4 insert一条数据
5 查询数据为100条
6 提交事务
7 查询数据为101条

我们可以看到,事务B在提交事务之前,事务A的两次查询结果是一致的。事务B提交事务以后,事务A再次查询,查询到了新增的这条数据。在事务A中,多次查询的结果不一致,这就是我们说的“不可重复读”。

Repeatable Read(可重读)

这是MySQL的默认事务隔离级别,它确保同一事务的多个实例在并发读取数据时,会看到同样的数据行。不过理论上,这会导致另一个棘手的问题:幻读 (Phantom Read)。简单的说,幻读指当用户读取某一范围的数据行时,另一个事务又在该范围内插入了新行,当用户再读取该范围的数据行时,会发现有新的“幻影” 行。

上面这一段是MySQL官方给出的解释,听着云里雾里。“可重读”这种隔离级别解决了上面例子中的问题,保证了同一事务内,多次查询的结果是一致的。也就是说,事务B插入数据提交事务后,事务A的查询结果也是100条,因为事务A在开启事务时,事务B插入的数据还没有提交。

但是,这又引出了另外一个情况,“幻读”。这个幻读我之前理解是有问题的,在面试时,被对方一顿质疑。现在我们就看看幻读的正确理解:

时间点 事务A 事务B
1 开启事务
2 开启事务
3 查询数据“张三”,不存在
4 插入数据“张三”
5 提交事务
6 查询数据“张三”,不存在
7 插入数据“张三”,不成功

事务A查询“张三”,查询不到,插入又不成功,“张三”这条数据就像幻觉一样出现。这就是所谓的“幻读”。网上对“幻读”还是其他的解释,都是错误的。比如像“幻读”和“不可重复读”是一样,只不过“幻读”是针对数据的个数。这些理解都是错误的。

Serializable(可串行化)

这是最高的隔离级别,它通过强制事务排序,使之不可能相互冲突,从而解决幻读问题。简言之,它是在每个读的数据行上加上共享锁。在这个级别,可能导致大量的超时现象和锁竞争。这种隔离级别很少使用,不给大家做过多的介绍了。

最新文章

  1. Spark Streaming源码解读之数据清理内幕彻底解密
  2. 2016 年青岛网络赛---Family View(AC自动机)
  3. js判断手机端Android手机还是iPhone手机
  4. Beginning Android 4 中 Demo Basic/Switch 的问题.
  5. 怎么用js代码改变单选框的选中状态
  6. css 动画 transform transition animation
  7. python之路第五篇之递归(进阶篇:续:经典例子剖析)
  8. Linux显示内存状态
  9. Python_FTP通讯软件
  10. Linux下C语言生成可执行文件的过程
  11. 服务器、应用框架、MVC、MTV
  12. Linux中文乱码 - - 更改Linux字符集
  13. SQL Server 深入解析索引存储(堆)
  14. BZOJ.1568.[JSOI2008]Blue Mary开公司(李超线段树)
  15. Eclipse 手动增加linker library
  16. Authorization With Pundit
  17. 数据导出之winform导出word(三)
  18. ISE14.7使用教程(一个完整工程的建立)
  19. hiho一下 第197周 逆序单词
  20. C#实现WinForm下DataGridView控件的拷贝和粘贴

热门文章

  1. Asp.Net Core 已支持 gRPC-Web !!
  2. LeetCode 第17题--电话号码的组合(DFS)
  3. Java入门 - 语言基础 - 04.对象和类
  4. 「 从0到1学习微服务SpringCloud 」06 统一配置中心Spring Cloud Config
  5. HTTP负责均衡模块(HTTP Upstream)
  6. linux之samba使用
  7. SVN: 聚合工程下的子工程无法使用 svn:ignore
  8. 在华为云上开启FTP服务并建立FTP站点来从本地向服务器发送和下载文件
  9. http请求中的 OPTIONS 多余请求消除,减少的案例
  10. Spring-cloud微服务实战【八】:API网关zuul