先锋之矛 — MySQL InnoDB 中的锁

640?wx_fmt=png

写在前面

数据库本质上是一种共享资源,因此在最大程度提供并发访问性能的同时,仍需要确保每个用户能以一致的方式读取和修改数据。锁机制(Locking)就是解决这类问题的最好武器。

首先新建表 test,其中 id 为主键,name 为辅助索引,address 为唯一索引。

CREATE TABLE `test` (  `id` int(11) NOT NULL AUTO_INCREMENT,  `name` int(11) NOT NULL,  `address` int(11) NOT NULL,  PRIMARY KEY (`id`),  UNIQUE KEY `idex_unique` (`address`),  KEY `idx_index` (`name`)) ENGINE=InnoDB AUTO_INCREMENT=7 DEFAULT CHARSET=utf8mb4;

INSERT 方法中的行锁

640?wx_fmt=jpeg

可见,如果两个事务先后对主键相同的行记录执行 INSERT 操作,因为事务 A 先拿到了行锁,事务 B 只能等待直到事务 A 提交后行锁被释放。同理,如果针对唯一索引字段 address 进行插入操作,也需要获取行锁,图同主键插入过程类似,不再重复。

但是,如果两个事务都针对辅助索引字段 name 进行插入,不需要等待获取锁,因为辅助索引字段即使值相同,在数据库中也是操作不同的记录行,不会冲突。

Update 方法与 Insert 方法结果类似。

SELECT FOR UPDATE 下的表锁与行锁

640?wx_fmt=jpeg

事务 A SELECT FOR UPDATE 语句会拿到表 test 的 Table Lock,此时事务 B 去执行插入操作会阻塞,直到事务 A 提交释放表锁后,事务 B 才能获取对应的行锁执行插入操作。

但是如果事务 A 的 SELECT FOR UPDATE 语句紧跟 WHERE id = 1 的话,那么这条语句只会获取行锁,不会是表锁,此时不阻塞事务 B 对于其他主键的修改操作

辅助索引下的间隙锁

先看下 test 表下的数据情况:

mysql> select * from test;+----+------+---------+| id | name | address |+----+------+---------+|  3 |    1 |       3 ||  6 |    1 |       2 ||  7 |    2 |       4 ||  8 |   10 |       5 |+----+------+---------+4 rows in set (0.00 sec)

间隙锁可以说是行锁的一种,不同的是它锁住的是一个范围内的记录,作用是避免幻读,即区间数据条目的突然增减。解决办法主要是:

  • 防止间隙内有新数据被插入,因此叫间隙锁

  • 防止已存在的数据,在更新操作后成为间隙内的数据(例如更新 id = 7 的 name 字段为 1,那么 name = 1 的条数就从 2 变为 3

InnoDB 自动使用间隙锁的条件为:

  • Repeatable Read 隔离级别,这是 MySQL 的默认工作级别

  • 检索条件必须有索引(没有索引的话会走全表扫描,那样会锁定整张表所有的记录)

当 InnoDB 扫描索引记录的时候,会首先对选中的索引行记录加上行锁,再对索引记录两边的间隙(向左扫描扫到第一个比给定参数小的值, 向右扫描扫描到第一个比给定参数大的值, 以此构建一个区间)加上间隙锁。如果一个间隙被事务 A 加了锁,事务 B 是不能在这个间隙插入记录的。

我们这里所说的 “间隙锁” 其实不是 GAP LOCK,而是 RECORD LOCK + GAP LOCK,InnoDB 中称之为 NEXT_KEY LOCK

下面看个例子,我们建表时指定 name 列为辅助索引,目前这列的取值有 [1,2,10]。间隙范围有 (-∞, 1]、[1,1]、[1,2]、[2,10]、[10, +∞)

Round 1:

  • 事务 A SELECT … WHERE name = 1 FOR UPDATE;

  • 对 (-∞, 2) 增加间隙锁

  • 事务 B INSERT … name = 1 阻塞

  • 事务 B INSERT … name = -100 阻塞

  • 事务 B INSERT … name = 2 成功

  • 事务 B INSERT … name = 3 成功

Round 2:

  • 事务 A SELECT … WHERE name = 2 FOR UPDATE;

  • 对 [1, 10) 增加间隙锁

  • 事务 B INSERT … name = 1 阻塞

  • 事务 B INSERT … name = 9 阻塞

  • 事务 B INSERT … name = 10 成功

  • 事务 B INSERT … name = 0 成功

Round 3:

  • 事务 A SELECT … WHERE name <= 2 FOR UPDATE;

  • 对 (-∞, +∞) 增加间隙锁

  • 事务 B INSERT … name = 3 阻塞

  • 事务 B INSERT … name = 300 阻塞

  • 事务 B INSERT … name = -300 阻塞

InnoDB 锁机制总结

640?wx_fmt=jpeg

参考资料

  • 《MySQL 技术内幕 InnoDB 存储引擎》第二版 姜承尧著

  • About MySQL InnoDB’s Lock

Python网络爬虫与推荐算法新闻推荐平台:网络爬虫:通过Python实现新浪新闻的爬取,可爬取新闻页面上的标题、文本、图片、视频链接(保留排版) 推荐算法:权重衰减+标签推荐+区域推荐+热点推荐.zip项目工程资源经过严格测试可直接运行成功且功能正常的情况才上传,可轻松复刻,拿到资料包后可轻松复现出一样的项目,本人系统开发经验充足(全领域),有任何使用问题欢迎随时与我联系,我会及时为您解惑,提供帮助。 【资源内容】:包含完整源码+工程文件+说明(如有)等。答辩评审平均分达到96分,放心下载使用!可轻松复现,设计报告也可借鉴此项目,该资源内项目代码都经过测试运行成功,功能ok的情况下才上传的。 【提供帮助】:有任何使用问题欢迎随时与我联系,我会及时解答解惑,提供帮助 【附带帮助】:若还需要相关开发工具、学习资料等,我会提供帮助,提供资料,鼓励学习进步 【项目价值】:可用在相关项目设计,皆可应用在项目、毕业设计、课程设计、期末/期/大作业、工程实训、大创等学科竞赛比赛、初期项目立项、学习/练手等方面,可借鉴此优质项目实现复刻,设计报告也可借鉴此项目,也可基于此项目来扩展开发出更多功能 下载后请首先打开README文件(如有),项目工程可直接复现复刻,如果基础还行,也可在此程序基础上进行修改,以实现其它功能。供开源学习/技术交流/学习参考,勿用于商业用途。质量优质,放心下载使用。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值