MySQL的innoDB存储引擎如何解决幻读的问题
基本情况
- MySQL有四种事务隔离级别,这四种隔离级别代表当存在多个事务并发冲突时,可能出现的脏读、不可重复读、幻读的问题
- InnoDB 在 RR 的隔离级别下 ,解决了幻读的问题
- 幻读是指在同一个事务中,前后两次查询相同的范围时,得到的结果数不一致
原理分析
- InnoDB 引入了间隙锁和 next-key Lock 机制来解决幻读问题
- 通过主键索引查询一条记录 ,并且对这条记录通过for update加锁,这个时候,会产生一个记录锁 ,也就是行锁
- 如果是范围查询 加 for update,其它事务执行 insert 语句,要被阻塞,直到前面获得锁的事务释放
- 在 InnoDB 中设计了一种间隙锁,它的主要功能是锁定一段范围内的索引记录
- 例如:当对查询范围 id > 4 and id < 7 加锁,会对B+树中 4, 7 的开区间索引加间隙锁,其他事务对这个区间的数据进行插入、更新、删除都会被锁住
- 如果查询范围是 id > 4,则会锁住多个索引区间,因此引入 next-key Lock 机制
- next-key Lock 相当于间隙锁和记录锁的合集,记录锁锁定存在的记录行 ,间隙锁锁住记录行之间的间隙 ,而 next-key Lock 锁住的是两者之和
临键锁
- 临键锁 也称为行级锁或行锁,是数据库管理系统中的一种锁机制,用于在并发环境下对数据库表中的行进行精确控制和保护
- 临键锁的作用范围是数据库表中的单行记录,相比表级锁或页级锁,临键锁的粒度更细,可以最大程度地提高并发性能
- 使用临键锁可能会发生死锁情况,即两个或多个事务互相等待对方释放锁,为了避免死锁,数据库管理系统通常会实现死锁检测和解决机制
不可重复读和幻读的区别是?
- 不可重复读 指的是在同一事务内,对于相同的查询条件,多次读取得到的结果不一致。这是由于其他并发事务在读取期间修改了数据造成的。例如,事务A首先读取某一行数据,然后事务B修改了该行数据,最后事务A再次读取同一行数据时,发现数据已经发生变化,导致结果不一致。
- **幻读 **则指的是在同一事务内,对于相同的查询条件,多次读取得到的结果集合大小不一致。这是由于其他并发事务在读取期间插入或删除了满足查询条件的数据造成的。例如,事务A首先执行了一个查询语句,返回了一组结果,接着事务B插入了满足相同查询条件的新数据,最后事务A再次执行相同的查询语句,发现结果集中新增了数据,导致结果集合大小不一致。
- 不可重复读主要涉及到同一行数据在事务内被修改,导致读取结果不一致;而幻读则涉及到符合查询条件的整个结果集合在事务内被插入或删除,导致结果集合大小不一致。