你应该了解的 MySQL 锁分类
MySQL 中的锁
锁是为了解决并发环境下资源竞争的手段,其中乐观并发控制,悲观并发控制和多版本并发控制是数据库并发控制主要采用的技术手段(具体可见我之前的文章),而 MySQL 中的锁就是其中的悲观并发控制。
MySQL 中的锁有很多种类,我们可以按照下面方式来进行分类。
按读写
从数据库的读写的角度来分,数据库的锁可以分为分为以下几种:
独占锁:又称排它锁、X 锁、写锁。X 锁不能和其他锁兼容,只要有事务对数据上加了任何锁,其他事务就不能对这些数据再放置 X 了,同时某个事务放置了 X 锁之后,其他事务就不能再加其他任何锁了,只有获取排他锁的事务是可以对数据进行读取和修改。
共享锁:又称读锁、S 锁。S 锁与 S 锁兼容,可以同时放置。
更新锁:又称 U 锁。它允许再加 S 锁,但不允许其他事务再施加 U 锁或 X 锁,当被读取的数据要被更新时,则升级 S 锁为 X 锁。U 锁的优点是允许事务 A 读取数据的同时不阻塞其它事务,并同时确保事务 A 自从上次读取数据后数据没有被更改,因此可以减少 X 锁和 S 锁的冲突,同时避免使用 S 锁后再升级为 X 锁造成的死锁现象。注意,MySQL 并不支持 U 锁,SQLServer 才支持 U 锁。
兼容性矩阵如下(\+ 代表兼容, -代表不兼容)
按粒度
MySQL 支持不同级别的锁,其锁定的数据的范围也不同,也即我们常说的锁的粒度。MySQL 有三种锁级别:行级锁、页级锁、表级锁。不同的存储引擎支持不同的锁粒度,例如 MyISAM 和 MEMORY 存储引擎采用的是表级锁,页级锁仅被 BDB 存储引擎支持,InnoDB 存储引擎支持行级锁和表级锁,默认情况下是采用行级锁。
特点
表级锁:开销小,加锁快;不会出现死锁;锁定粒度大,发生锁冲突的概率最高,并发度最低。数据库引擎总是一次性同时获取所有需要的锁以及总是按相同的顺序获取表锁从而避免死锁。
行级锁:开销大,加锁慢;会出现死锁;锁定粒度最小,发生锁冲突的概率最低,并发度也最高。行锁总是逐步获得的,因此会出现死锁。
页面锁:开销和加锁时间界于表锁和行锁之间;会出现死锁;锁定粒度界于表锁和行锁之间,并发度一般。
下面详细介绍行锁和表锁,页锁由于使用得较少就不介绍了。
行锁
按行对数据进行加锁。InnoDB 行锁是通过给索引上的索引项加锁来实现的,Innodb 一定存在聚簇索引,行锁最终都会落到聚簇索引上,通过非聚簇索引查询的时候,先锁非聚簇索引,然后再锁聚簇索引。如果一个 where 语句里面既有聚簇索引,又有二级索引,则会先锁聚簇索引,再锁二级索引。由于是分步加锁的,因此可能会有死锁发生。
MySQL 的行锁对 S、X 锁上做了一些更精确的细分,使得行锁的粒度更细小,可以减少冲突,这就是被称为“precise mode”的兼容矩阵。(该矩阵没有出现在官方文档上,是有人通过 Mysql lock0lock.c:lockrechas_to_wait 源代码推测出来的。)
行锁兼容矩阵
间隙锁(Gap Lock):只锁间隙,前开后开区间(a,b),对索引的间隙加锁,防止其他事务插入数据。
记录锁(Record Lock):只锁记录,特定几行记录。
临键锁(Next-Key Lock):同时锁住记录和间隙,前开后闭区间(a,b]。
插入意图锁(Insert Intention Lock):插入时使用的锁。在代码中,插入意图锁,实际上是 GAP 锁上加了一个 LOCKINSERTINTENTION 的标记。
S 锁和 S 锁是完全兼容的,因此在判别兼容性时不需要对比精确模式。精确模式的检测,用在 S、X 和 X、X 之间。从这个矩阵可以看到几个特点:
INSERT 操作之间不会有冲突:你插入你的,我插入我的。
GAP,Next-Key 会阻止 Insert:插入的数据正好在区间内,不允许插入。
GAP 和 Record,Next-Key 不会冲突
Record 和 Record、Next-Key 之间相互冲突。
已有的 Insert 锁不阻止任何准备加的锁。
间隙锁(无论是 S 还是 X)只会阻塞 insert 操作。
注意点
对于记录锁,列必须是唯一索引列或者主键列,查询语句必须为精确匹配,如“=”,否则记录锁会退化为临键锁。
间隙锁和临键锁基于非唯一索引,在唯一索引列上不存在间隙锁和临键锁。
表锁与锁表的误区
只有正确通过索引条件检索数据(没有索引失效的情况),InnoDB 才会使用行级锁,否则 InnoDB 对表中的所有记录加锁,也就是将锁住整个表。注意,这里说的是锁住整个表,但是 Innodb 并不是使用表锁来锁住表的,而是使用了下面介绍的 Next-Key Lock 来锁住整个表。网上很多的说法都是说用表锁,然而实际上并不是,我们可以通过下面的例子来看看。
假设我们有以下的数据(MySQL8):
方法一:
我们使用表锁锁表,并查看引擎的状态
然后我们再通过非索引的字段查询来加锁,并查看引擎的状态
然后我们再删除 id 为 2,3,4 的数据,然后在通过非索引的字段查询来加锁,并查看引擎的状态
可以看到这里使用了表锁和因为没法用索引锁定特定行而转而锁住整个表是不一样的。从第二次和第三次的操作来看,lock 住的 row 也是不同的,这是因为两者间隙的个数不同,所以可以看到使用的并不是表锁,而是 Next-Key Lock。第一次锁住了(-∞,1],(1,2],(2,3],(3,4],(4,5],(5,∞],第二次锁住了(-∞,1],(1,5],(5,∞]。
方法二:
也可以通过以下语句来查看锁的信息,也可以知道用的是行锁,且是锁住了区间(插入不了数据)和记录,所以是 Next-Key Lock。
LOCK_TYPE:对于 InnoDB,可选值为 RECORD(行锁), TABLE(表锁)
LOCKMODE:对于 InnoDB,可选值为 S[,GAP], X[,GAP], IS[,GAP],IX[,GAP], AUTOINC 和 UNKNOWN。除了 AUTO_INC 和 UNKNOWN,其他锁定模式都包含了 GAP 锁(如果存在)。
具体可见 MySQL 文档:https://dev.mysql.com/doc/refman/8.0/en/performance-schema-data-locks-table.html
表级锁
直接对整个表加锁,影响表中所有记录,表读锁和表写锁的兼容性见上面的分析。
MySQL 中除了表读锁和表写锁之外,还存在一种特殊的表锁:意向锁,这是为了解决不同粒度的锁的兼容性判断而存在的。
意向锁
因为锁的粒度不同,表锁的范围覆盖了行锁的范围,所以表锁和行锁会产生冲突,例如事务 A 对表中某一行数据加了行锁,然后事务 B 想加表锁,正常来说是应该要冲突的。如果只有行锁的话,要判断是否冲突就得遍历每一行数据了,这样的效率实在不高,因此我们就有了意向表锁。
意向锁的主要目的是为了使得 行锁 和 表锁 共存,事务在申请行锁前,必须先申请表的意向锁,成功后再申请行锁。注意:申请意向锁的动作是数据库完成的,不需要开发者来申请。
意向锁是表级锁,但是却表示事务正在读或写某一行记录,而不是整个表, 所以意向锁之间不会产生冲突,真正的冲突在加行锁时检查。
意向锁分为意向读锁(IS)和意向写锁(IX)。
表锁的兼容性矩阵
参考资料
版权声明: 本文为 InfoQ 作者【X先生】的原创文章。
原文链接:【http://xie.infoq.cn/article/4cd6a853cbf8951daa8c5bda9】。文章转载请联系作者。
评论