MySQL 幻读

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

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

幻读的影响

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


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

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

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

如何解决幻读

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

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

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

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

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

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

你可能感兴趣的:(MySQL 幻读)