当前位置:网站首页>The soul asks: How does MySQL solve phantom reads?
The soul asks: How does MySQL solve phantom reads?
2022-08-01 14:17:00 【Java Notes Shrimp】
点击关注公众号,利用碎片时间学习
概念
MySQL InnoDB支持三种行锁定方式:
行锁(Record Lock):锁直接加在索引记录上面.
间隙锁(Gap Lock):锁加在不存在的空闲空间,可以是两个索引记录之间,也可能是第一个索引记录之前或最后一个索引之后的空间.
Next-Key Lock:行锁与间隙锁组合起来用就叫做
Next-Key Lock.
当我们用范围条件而不是相等条件检索数据,并请求共享或排他锁时,InnoDB会给符合条件的已有数据记录的索引项加锁;对于键值在条件范围内但并不存在的记录,叫做“间隙(GAP)”,InnoDB也会对这个“间隙”加锁,这种锁机制就是所谓的间隙锁(Next-Key锁).
举例来说,假如user表中只有101条记录,其empid的值分别是 1,2,...,100,101,下面的SQL:
select * from user where user_id > 100 for update;是一个范围条件的检索,InnoDB不仅会对符合条件的user_id值为101的记录加锁,也会对user_id大于101(这些记录并不存在)的“间隙”加锁.
产生幻读的原因是,行锁只能锁住行,但是新插入记录这个动作,要更新的是记录之间的“间隙”.因此,为了解决幻读问题,InnoDB 只好引入新的锁,也就是间隙锁 (Gap Lock).
InnoDB使用间隙锁的目的,一方面是为了防止幻读,以满足相关隔离级别的要求,对于上面的例子,要是不使用间隙锁,如果其他事务插入了user_id大于100的任何记录,那么本事务如果再次执行上述语句,就会发生幻读;另外一方面,是为了满足其恢复和复制的需要
快照读和当前读
快照读历史数据-mvcc
innodb的默认事务隔离级别是rr(可重复读).它的实现技术是mvcc(MVCC只在读提交可重复读两种隔离级别下工作).基于版本的控制协议.该技术不仅可以保证innodb的可重复读,而且可以防止幻读.但是它防止的是快照读,也就是读取的数据虽然是一致的,但是数据是历史数据.
当前读最新数据-next-key lock
如何做到保证数据是一致的(也就是一个事务,其内部读取对应某一个数据的时候,数据都是一样的),同时读取的数据是最新的数据.innodb提供了next-key lock,也就是结合gap锁与行锁,达到最终目的.
实现:
1.快照读(snapshot read)
简单的select操作(不包括 select ... lock in share mode, select ... for update)
2.当前读(current read)
select ... lock in share mode、select ... for update
insert、update、delete在RR级别下,快照读是通过MVCC(多版本控制)和undo log来实现的,当前读是通过加record lock(记录锁)和gap lock(间隙锁)来实现的.
测试
建表innodb_lock:
DROP TABLE IF EXISTS `innodb_lock`;
CREATE TABLE `innodb_lock` (
`a` int(10) NOT NULL,
`b` varchar(255) NOT NULL DEFAULT '',
KEY `index_a` (`a`),
KEY `index_b` (`b`)
) ENGINE=InnoDB;插入数据,注意这里边没有a为2的数据:
INSERT INTO `innodb_lock` VALUES ('1', 'b2');
INSERT INTO `innodb_lock` VALUES ('3', '3');
INSERT INTO `innodb_lock` VALUES ('4', '4000');
INSERT INTO `innodb_lock` VALUES ('5', '5000');
INSERT INTO `innodb_lock` VALUES ('6', '6000');
INSERT INTO `innodb_lock` VALUES ('7', '7000');
INSERT INTO `innodb_lock` VALUES ('8', '8000');
INSERT INTO `innodb_lock` VALUES ('9', '9000');(1)开启两个客户端,修改事务隔离级别为可重复读
(2)开启事务,在左侧客户端批量修改a为1~6范围内的数据.在右侧客户端插入a为2的数据.右侧操作被阻塞.说明有间隙锁.
(3)重复(2),事务隔离级别依然是repeatable read,只不过变成在右侧客户端插入a为10的数据,成功.
(4)事务隔离级别设置为read committed,重复步骤(2),发现右侧客户端的操作成功,说明该隔离级别无间隙锁.
(5)还要特别说明的是,InnoDB除了通过范围条件加锁时使用间隙锁外,如果使用相等条件请求给一个不存在的记录加锁,InnoDB也会使用间隙锁!左侧客户端给不存在的记录加锁,右侧客户端的增加操作阻塞.
但是,如果a是唯一索引,不会升级全表锁.
先添加唯一索引:

(6)重复步骤(5),发现右侧客户端不会被阻塞,数据插入成功

小结
很显然,在使用范围条件检索并锁定记录时,InnoDB这种加锁机制会阻塞符合条件范围内键值的并发插入,这往往会造成严重的锁等待.因此,在实际应用开发中,尤其是并发插入比较多的应用,我们要尽量优化业务逻辑,尽量使用相等条件来访问更新数据,避免使用范围条件;当然,对一条不存在的记录加锁,也会有间隙锁的问题.
间隙锁在InnoDB的唯一作用就是防止其它事务的插入操作,以此来达到防止幻读的发生,所以间隙锁不分什么共享锁与排它锁.如果InnoDB扫描的是一个主键、或是一个唯一索引的话,那InnoDB只会采用行锁方式来加锁,而不会使用Next-Key Lock的方式,也就是说不会对索引之间的间隙加锁,仔细想想的话,这个并不难理解,大家也可以自己测试一下.
要禁止间隙锁的话,可以把隔离级别降为读已提交,或者开启参数innodb_locks_unsafe_for_binlog.
补充
MySQL InnoDB的可重复读并不保证避免幻读,需要应用使用加锁读来保证.而这个加锁度使用到的机制就是next-key locks.
本事务中第一次读取出一行,做了一次更新后,另一个事务里提交的数据就出现了,也可以看做是一种幻读:
t Session A Session B
|
| START TRANSACTION; START TRANSACTION;
|
| SELECT * FROM innodb_lock;
| +------+-------+
| | a | b |
| +------+-------+
| | 1 | a |
| +------+-------+
| INSERT INTO innodb_lock
| VALUES (2, 'b');
|
| SELECT * FROM innodb_lock;
| +------+-------+
| | a | b |
| +------+-------+
| | 1 | a |
| +------+-------+
| COMMIT;
|
| SELECT * FROM innodb_lock;
| +------+-------+
| | a | b |
| +------+-------+
| | 1 | a |
| +------+-------+
|
| UPDATE innodb_lock SET b='z';
| Rows matched: 2 Changed: 2 Warnings: 0
| (怎么多出来一行)
|
| SELECT * FROM innodb_lock;
| +------+-------+
| | a | b |
| +------+-------+
| | 1 | z |
| | 2 | z |
| +------+-------+
|来源:blog.csdn.net/sinat_27143551/article/
details/81736330
推荐:
最全的java面试题库
PS:因为公众号平台更改了推送规则,如果不想错过内容,记得读完点一下“在看”,加个“星标”,这样每次新文章推送才会第一时间出现在你的订阅列表里.点“在看”支持我们吧!边栏推荐
- 什么是闭包?
- Yann LeCun开怼谷歌研究:目标传播早就有了,你们创新在哪里?
- redis主从同步方式(redis数据同步原理)
- 【码蹄集新手村600题】判断一个数字是否为完全平方数
- 论文详读《基于改进 LeNet-5 模型的手写体中文识别》,未完待补充
- AtCoder Beginner Contest 261 D - Flipping and Bonus
- 2022-07-29 网工进阶(二十二)BGP-其他特性(路由过滤、团体属性、认证、AS欺骗、对等体组、子路由器、路由最大接收数量)
- 从零开始Blazor Server(4)--登录系统
- Service Mesher Meetup 成都站:Service Mesh是下一代SDN吗?
- 易优压双驱挖掘机压路机器类网站源码 v1.5.8
猜你喜欢

Wovent Bio IPO: Annual revenue of 480 million pension fund is a shareholder

Longkou united chemical registration: through 550 million revenue xiu-mei li control 92.5% stake

PAT 1162 Postfix Expression(25)

易优压双驱挖掘机压路机器类网站源码 v1.5.8

立新能源深交所上市:市值55亿 哈密国投与国有基金是股东

【码蹄集新手村600题】判断一个数字是否为完全平方数

Batch replace tables in Word with pictures and save

Gradle series - Gradle tests, Gradle life cycle, settings.gradle description, Gradle tasks (based on Groovy documentation 4.0.4) day2-3

HTB-Mirai

免费使用高性能的GPU和TPU—谷歌Colab使用教程
随机推荐
the direction i'm looking for
荣信文化通过注册:年营收3.8亿 王艺桦夫妇为实控人
阿里巴巴测试开发岗P6面试题
大佬们,datax同步数据,同步过程中要新增一个uuid,请问column 怎么写pgsql,uu
What Can Service Mesh Learn from SDN?
Two Permutations
超全!全国近90所大学考研报录比汇总!
龙口联合化学通过注册:年营收5.5亿 李秀梅控制92.5%股权
34、树莓派进行人体姿态检测并进行语音播报
What is a closure?
微信UI在线聊天源码 聊天系统PHP采用 PHP 编写的聊天软件,简直就是一个完整的迷你版微信
搭建ntp时间服务器(安装sql2000配置服务器失败)
倪光南:openEuler已达国际同类社区水准
透过现象看本质,如何针对用户做好需求分析
Qt实战案例(55)——利用QDir删除选定文件目录下的空文件夹
leetcode.26 删除有序数组中的重复项(set/直接遍历)
DDL和DML的含义与区别「建议收藏」
math.pow()函数用法[通俗易懂]
iPhone难卖,被欧洲反垄断的服务业务也难赚钱了,苹果的日子艰难
HTB-Mirai