MySQL 幻读

MySQL 在当前读的情况下,如果看到了其他事务新插入的一条数据,这种情况称为幻读。
是指一个事务在前后两次查询同一个范围的时候,后一次查询看到了前一次查询没有看到的行。

  1. 可重复读 的隔离级别下,普通的查询时快照读,是不会看到别的事务插入的数据的。因此,幻读在“当前读” 下才会出现。
  2. 幻读仅指新插入的行。

幻读的影响

语义上的影响。例如下图:


sessionA 声明了“我要把所有 d=5 的行锁住,不准别的事务进行再进行操作”。而实际上,这个语义被破坏了。

数据一致性的问题。锁的设计师为了保证数据的一致性,一致性不只是数据库内部数据状态的一致性,还包含了数据和日志在逻辑上的一致性。

如果出现幻读,数据库的binlog 的顺序就会错乱,因为 binlog 是在事务提交之后记录的。如果使用了这样的binlog 去备库执行或是克隆一个库,就会出问题。

如何解决幻读

为了解决幻读,InnoDB 引入了新的锁,就是间隙锁(gap lock)。

间隙锁就是在执行当前读的时候,不止是给数据库已有的数据加上行锁,还同时加了间隙锁,确保无法插入新的记录。

间隙锁只跟“往这个间隙中插入一条记录”这个操作冲突。间隙锁之间不存在冲突。

间隙锁与行锁并称为 next-key lock。

但是间隙锁的引入,可能会导致同样的语句锁住了更大的范围,这其实是影响了并发度的。如果两个事务同时插入一条数据,而且持有同样的间隙锁,就会产生死锁。

有的公司会使用读提交的隔离级别,然后把 binlog 的格式设置为 row,也可以解决间隙锁会产生死锁的问题。但是具体的业务也需要具体分析。

©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • 幻读(phantom read) 前提条件:InnoDB引擎,可重复读隔离级别,使用当前读时。 表现:一个事务(同...
    慧鑫coming阅读 69,858评论 6 36
  • 1.什么是幻读幻读,并不是说两次读取获取的结果集不同,幻读侧重的方面是某一次的 select 操作得到的结果所表征...
    czcphp阅读 1,904评论 1 3
  • 事务可以用来维护数据库的完整性,保证成批的 SQL 语句要么全部执行,要么全部不执行。MySQL 中只有使用了 I...
    伊凡的一天阅读 2,398评论 3 22
  • 幻读 首先我们要搞明白何谓幻读,目前网上的众多解释幻读的博文个人感觉仔细设想一下就能找出推翻的例子,就像博文把 非...
    小陈阿飞阅读 2,022评论 0 1
  • 缓存的场景各种各样,稍微一改代码就大不相同。所以我写的只是这一种场景:缓存一个页面(只是一个页面!),并且记录下这...
    小北呀_阅读 391评论 0 0