1.什么样的状况叫作幻读?
2.Mysql 可重复读隔离级别下,到底能不能阻止幻读?
3.什么是当前读,什么是快照读?sql
事务A 按照必定条件进行数据读取, 期间事务B 插入了相同搜索条件的新数据,事务A再次按照原先条件进行读取时,发现了事务B 新插入的数据 称为幻读
并发
若是事务A 按必定条件搜索, 期间事务B 删除了符合条件的某一条数据,致使事务A 再次读取时数据少了一条。这种状况归为 不可重复读ui
Mysql 隔离级别为 可重复读
;spa
CREATE TABLE `author` (
`id` int(11) NOT NULL AUTO_INCREMENT,
`name` varchar(255) DEFAULT NULL,
`age` int(11) DEFAULT NULL,
PRIMARY KEY (`id`)
) ENGINE=InnoDB AUTO_INCREMENT=0 DEFAULT CHARSET=utf8;
INSERT into author VALUES (1,'g1',20),(5,'g5',20),(15,'g15',30),(20,'g20',30);
复制代码
时间 | Session 1 | Session2 |
---|---|---|
begin; | ||
T1 | select * from author where age = 20; 结果: (1,g1,20),(5,g5,20) |
|
T2 | INSERT into author VALUES (25,'g25',20); | |
T3 | select * from author where age = 20; 结果: (1,g1,20),(5,g5,20) |
|
T4 | update author set name = 'G0' where age = 20 结果: 显示影响行数为3行。 |
|
T5 | select * from author where age = 20; 结果:(1,g0,20),(5,g0,20),(25,g0,20) |
来分析下情形:code
貌似
Session2新插入的数据并未影响到Session1的事务读取。对于T1 -- T3 时刻的情形,从结果来看,在
可重复度读
隔离级别下彷佛解决了幻读
的问题。进程
T4,T5 的结果来看,Session1 读到了 Session2 新插入的数据。产生了
幻读
现象事务
了解过MVCC的同窗,确定知道或据说过当前读
,和快照读。
(不知道的同窗,能够查找相关资料了解下,固然后续我也会有文章专门介绍MVCC)。
首先要知道的是MVCC 就InnoDB 秒级创建数据快照的能力。 快照读
就是读取数据的时候会根据必定规则读取事务可见版本的数据。 而当前读
就是读取最新版本的数据。
什么状况下使用的是快照读:(快照读,不会加锁)string
通常的 select * from .... where ... 语句都是快照读it
什么状况下使用的是当前读:(当前读,会在搜索的时候加锁)io
select * from .... where ... for update select * from .... where ... lock in share mode update .... set .. where ... delete from. . where ..
若是事务中都使用快照读,那么就不会产生幻读现象,可是快照读和当前读混用就会产生幻读。
先让咱们数据恢复到初始状态
TRUNCATE TABLE author;
INSERT into author VALUES (1,'g1',20),(5,'g5',20),(15,'g15',30),(20,'g20',30);
复制代码
时间 | Session 1 | Session2 |
---|---|---|
begin; | ||
T1 | select * from author where age = 20 for update; 结果: (1,g1,20),(5,g5,20) |
|
T2 | INSERT into author VALUES (25,'g25',20); 阻塞,等待锁 |
|
T3 | select * from author where age = 20 for update; 结果: (1,g1,20),(5,g5,20) |
|
能够看到Session 2 被阻塞了。须要等到Session1 提交事务后才能完成。当咱们在事务中每次读取都使用当前读
,也就是人工把InnoDB变成了串行化
。必定程度上下降了并发性,可是也一样避免了幻读的状况。
当前读为何会阻塞新数据的插入,主要是间隙锁
的加锁机制。(Mysql 的锁
也是个大问题,后续有专门的章节介绍)
新插入的数据
,这种现象叫幻读。