美文网首页
MySQL可重复读防止幻读

MySQL可重复读防止幻读

作者: 西5d | 来源:发表于2018-03-26 10:31 被阅读1461次

接上篇事务隔离级别和幻读,留了个坑,没想到竟然过了10天,时间不注意真的过的好快。顺便提下,简书图片链接是属于网站的,开发自己的图床迫在眉睫,万一哪天迁移就要做很多额外工作,一些概念或者思路用图片表达更直观清楚。
回到正题,之前提到一般情况下MySQL的InnoDB引擎在可重复读的情况下是没法保证不出现幻读的,但实际情况是MySQL可以通过加锁来防止幻读的出现,这种锁定通过Next-key机制来实现,是属于记录锁和间隙锁(Gap锁)的结合。
引申,行级别锁的三种算法:

  • Record lock
  • Gap lock
  • Next-Key lock
    record lock 按照索引记录加锁,如果没有则采用隐式的主键来锁定。
    next-key 结合了Gap lock 和 record lock ,例如索引值:10,11,13,20。则按next-key 锁定的区间为(-∞,10],(10,11]
    ,(11,13],(13,20],(20,+∞)

注意此处指非唯一索引,如果是唯一索引,会降级为Record lock ,仅仅锁住索引本身,而非范围。同时这种降级只发生在查询所有索引列的情况下,如果存在联合索引且只是查询联合索引的某一列,即属于range类型查询,仍是采用Next-key锁定,不会降级。

举个存在唯一索引和辅助索引的例子做说明:

create table test ( a int , b int , primary key (a), key(b));
insert into test select 1, 1;
insert into test select 3, 1;
insert into test select 5, 3;
insert into test select 7, 6;
insert into test select 10,8;

执行 select * from test where b = 3 for update
存在两个索引,分别加锁,唯一主键列a加record lock , 辅助索引列b加next-key lock (1,3) 以及给下一个值的区间(3,6)加gap锁;
因此在另一个事务里执行以下语句都会阻塞,具体分析:

select * from test where a = 5 lock in share mode;
insert into test select 4,2;
insert into test select 6,5;

第一个阻塞因为加了唯一索引的record lock a = 5;
第二个主键插入4,符合条件,但是根据辅助索引b 的范围, b = 2 在(1,3)中,同样阻塞;
第三个a =6 不在主键a锁定范围,b = 5 也不在辅助索引b 的范围(1,3)中,但在另一个gap锁范围(3,6)中,因此也阻塞;
这种锁定情形下,可以执行的包括类似语句:

insert into test select 8,6; 
insert into test select 2,0;

gap lock 可以通过设置隔离级别为读已提交或者将innodb_locks_unsafe_for_binlog = 1取消,但是从隔离性,性能,以及主从数据一致等方面都不建议这样做。

insert的特殊情况
对于insert 会检查下一条记录是否被锁定,如上述例子有select * from test where b = 3 for update插入insert into test select 2,2会检测到b = 3 已经被锁定,而insert into test select 2,0可以执行;

[1]:《MySQL技术内幕:InnoDB存储引擎》-第六章:锁

相关文章

  • MySQL可重复读防止幻读

    接上篇事务隔离级别和幻读,留了个坑,没想到竟然过了10天,时间不注意真的过的好快。顺便提下,简书图片链接是属于网站...

  • MySQL的可重复读级别能解决幻读吗

    转载 MySQL的可重复读级别能解决幻读吗MySQL事务隔离级别的实现原理

  • MYSQL -锁机制01

    一、前提1、MySQL版本:5.7.192、引擎:InnoDB3、隔离机制:可重复读。 二、不可重复读与幻读的区别...

  • spring事务整理

    隔离级别 1、默认使用数据库的(mysql:可重复读,oracle:已提交读) 2、未提交读 会出现脏读、幻读、不...

  • mysql知识点

    mysql概览 一些基本问题 事务 事务相关基本问题 脏读 丢失修改 不可重复读 幻读 不可重复读 vs 幻读不可...

  • 66 MVCC 多版本控制模型原理与间隙锁,悲观锁。乐观锁

    1 MYSQL 事务隔离级别 四种隔离级别: 脏读,可重复读,幻读总结2,详细画图分析 MVCC 多版本并发控制与...

  • mysql 解决可提交读、可重复读、幻读

    1.重要的概念 mysql的锁有哪些 这张图本人觉得总结得挺好的,在一般的互联网项目中,基本上用的都是Innodb...

  • InnoDB的MVCC如何解决幻读

    InnoDB默认的隔离级别是RR(可重复读),可以解决脏读和不可重复读,但是不能解决幻读问题。 什么是幻读? 事务...

  • 1.4 MVCC多版本控制的实现

    前面提到的在mysql中事务的级别默认是repeatable read(可重复读),其有出现幻读的可能性,即事务A...

  • Mysql MVCC机制

    概述 事务的ACID特性里面,Mysql通过MVCC机制来保证隔离性,实现可重复读的特性,但是可能会出现幻读的情况...

网友评论

      本文标题:MySQL可重复读防止幻读

      本文链接:https://www.haomeiwen.com/subject/vmahcftx.html