彻底搞懂 MySQL 中的锁机制
锁(Locking)是数据库在并发访问时保证数据一致性和完整性的主要机制。在 MySQL 中,不同存储引擎使用不同的加锁方式;我们以 InnoDB 存储引擎为例介绍 MySQL 中的锁机制,其他存储引擎中的锁相对简单一些。
一、锁的分类
1、表级锁与行级锁
MySQL 中的锁可以按照粒度分为锁定整个表的表级锁(table-level locking)和锁定数据行的行级锁(row-level locking):
- 表级锁具有开销小、加锁快的特性;表级锁的锁定粒度较大,发生锁冲突的概率高,支持的并发度低;
- 行级锁具有开销大,加锁慢的特性;行级锁的锁定粒度较小,发生锁冲突的概率低,支持的并发度高。
InnoDB 存储引擎同时支持行级锁(row-level locking)和表级锁(table-level locking),默认情况下采用行级锁。
表级锁适用于并发较低、以查询为主的应用,例如中小型的网站;MyISAM 和 MEMORY 存储引擎采用表级锁。
行级锁适用于按索引条件高并发更新少量不同数据,同时又有并发查询的应用,例如 OLTP 系统;InnoDB 和 NDB 存储引擎实现了行级锁。
2、共享锁与排他锁
InnoDB 实现了以下两种类型的行锁:
- 共享锁(S):允许获得该锁的事务读取数据行(读锁),同时允许其他事务获得该数据行上的共享锁,并且阻止其他事务获得数据行上的排他锁。
- 排他锁(X):允许获得该锁的事务更新或删除数据行(写锁),同时阻止其他事务取得该数据行上的共享锁和排他锁。
这两种行锁之间的兼容性如下:
共享锁和共享锁可以兼容,排他锁和其它锁都不兼容。例如,事务 A 获取了一行数据的共享锁,事务 B 可以立即获得该数据行的共享锁,也就是锁兼容;但是此时事务 B 如果想获得该数据行的排他锁,则必须等待事务 A 释数据行上的共享锁,此种情况存在锁冲突。
默认情况下,数据库中的锁都可以自动获取;但是也可以手动为数据进行加锁。我们来看一个示例,首先创建一个表:
create table t(
id int auto_increment primary key,
c1 int,
c2 int,
c3 int
);
create unique index idx_t_c1 on t(c1);
create index idx_t_c2 on t(c2);
insert into t(c1,c2,c3) values (1,1,1),(2,3,4),(3,6,9);
其中,id 是主键;c1 上创建了一个唯一索引;c2 上创建了一个非唯一索引;c3 上没有索引。
接下来的示例都使用 MySQL 默认的隔离级别 Repeatable Read,除非另有说明。
然后创建两个数据库连接 T1 和 T2,先在 T1 中锁定一行数据:
-- T1
mysql> begin;
Query OK, 0 rows affected (0.00 sec)
mysql> select * from t where id = 1 for share;
+----+------+------+------+
| id | c1 | c2 | c3 |
+----+------+------+------+
| 1 | 1 | 1 | 1 |
+----+------+------+------+
1 row in set (0.00 sec)
我们在事务中使用select ... for share
语句获得了数据行 id = 1 上的共享锁;对于 MySQL 8.0 之前的版本,可以使用select ... lock in share mode
命令。
由于 InnoDB 中的自动提交 autocommit 默认设置为 ON,我们必须在事务中为数据行加锁;或者将 autocommit 设置为 OFF。
然后在 T2 中执行以下语句:
-- T2
mysql> select * from t where id = 1 for share;
+----+------+------+------+
| id | c1 | c2 | c3 |
+----+------+------+------+
| 1 | 1 | 1 | 1 |
+----+------+------+------+
1 row in set (0.00 sec)
结果显示,在 T2 中成功获取改行数据上的共享锁。然后尝试获取排他锁:
-- T2
mysql> select * from t where id = 1 for update;
ERROR 1205 (HY000): Lock wait timeout exceeded; try restarting transaction
使用select ... for update
命令获取排他锁,此时该命令会一直处于等待状态并且最终超时。也就是说,共享锁和排他锁不兼容。
最后,在 T1 中提交或者回滚事务:
-- T1
mysql> commit;
3、意向锁
InnoDB 除了支持行级锁,还支持由 MySQL 服务层实现的表级锁(LOCK TABLES ... WRITE在指定的表加上表级排他锁)。当这两种锁同时存在时,可能导致冲突。例如,事务 A 获取了表中一行数据的读锁;然后事务 B 申请该表的写锁(例如修改表的结构)。如果事务 B 加锁成功,那么它就应该能修改表中的任意数据行,但是 A 持有的行锁不允许修改锁定的数据行。显然数据库需要避免这种问题,B 的加锁申请需要等待 A 释放行锁。
那么如何判断事务 B 是否应该获取表级锁呢?首先需要看该表是否已经被其他事务加上了表级锁,然后依次查看该表中的每一行是否已经被其他事务加上了行级锁。这种方式需要遍历整个表中的记录,效率很低。为此,InnoDB 引入了另外一种锁:意向锁(Intention Lock)。
意向锁属于表级锁,由 InnoDB 自动添加,不需要用户干预。意向锁也分为共享和排他两种方式:
- 意向共享锁(IS):事务在给数据行加行级共享锁之前,必须先取得该表的 IS 锁。
- 意向排他锁(IX):事务在给数据行加行级排他锁之前,必须先取得该表的 IX 锁。
此时,事务 A 必须先申请该表的意向共享锁,成功后再申请数据行的行锁。事务 B 申请表锁时,数据库查看该表是否已经被其他事务加上了表级锁;如果发现该表上存在意向共享锁,说明表中某些数据行上存在共享锁,事务 B 申请的写锁会被阻塞。
因此,意向锁是为了使得行锁和表锁能够共存,从而实现多粒度的锁机制。以下是表级锁和表级意向锁的兼容性:
InnoDB 表存在两种表级锁,一种是
LOCK TABLES
语句手动指定的锁,另一种是由 InnoDB 自动添加的意向锁。
简单来说,意向锁和表锁之间只有共享锁兼容,意向锁和意向锁之间都可以兼容。意向锁的主要作用是表明某个事务正在或者即将锁定表中的数据行。
我们以意向排他锁 IX 为例,继续上面的实验。先在 T1 中执行以下加锁语句:
-- T1
mysql> begin;
Query OK, 0 rows affected (0.00 sec)
mysql> select * from t where id = 1 for update;
+----+------+------+------+
| id | c1 | c2 | c3 |
+----+------+------+------+
| 1 | 1 | 1 | 1 |
+----+------+------+------+
1 row in set (0.00 sec)
在事务中为表 t 中的数据行 id = 1 加上了排他锁,同时会为表 t 加上意向排他锁。然后在 T2 中执行以下语句:
-- T2
mysql> lock tables t read; -- lock tables t write;
lock tables ... read
语句用于为表 t 加上表级共享锁;因为意向排他锁和表级共享锁冲突,所以 T2 一直等待 T1 释放锁。
也可以使用
lock tables ... write
语句为表 t 加上表级排他锁;因为意向排他锁和表级排他锁冲突,所以该语句也会一直等待 T1 释放锁。
当我们在 T1 中提交或者回滚事务:
-- T1
mysql> commit;
Query OK, 0 rows affected (0.00 sec)
T2 自动获得该表上的共享锁:
-- T2
Query OK, 0 rows affected (1 min 43.17 sec)
mysql> unlock tables;
Query OK, 0 rows affected (0.00 sec)
以上的unlock tables
语句用于释放该表上的排他锁。
我们再来验证一下两个意向排他锁之间锁的兼容性,先在 T1 中执行以下加锁语句:
-- T1
mysql> begin;
Query OK, 0 rows affected (0.00 sec)
mysql> select * from t where id = 1 for update;
+----+------+------+------+
| id | c1 | c2 | c3 |
+----+------+------+------+
| 1 | 1 | 1 | 1 |
+----+------+------+------+
1 row in set (0.00 sec)
T1 为表 t 加上了意向排他锁和数据行 id = 1 上的排他锁。然后在 T2 中执行以下语句:
-- T2
mysql> select * from t where id = 2 for update;
+----+------+------+------+
| id | c1 | c2 | c3 |
+----+------+------+------+
| 2 | 2 | 3 | 4 |
+----+------+------+------+
1 row in set (0.00 sec)
T2 成功为数据行 id = 2 加上的排他锁,同时为表 t 加上了意向排他锁。也就是说,T1 和 T2 同时获得了表 t 上的意向排他锁,以及不同数据行上的行级排他锁。InnoDB 通过行级锁,实现了更细粒度的控制,能够支持更高的并发更新和查询。
最后在 T1 中提交或者回滚事务:
-- T1
mysql> commit;
Query OK, 0 rows affected (0.00 sec)
二、行级锁实现
InnoDB 通过给索引上的索引记录加锁的方式实现行级锁。具体来说,InnoDB 实现了三种行锁的算法:记录锁(Record Lock)、间隙锁(Gap Lock)和 Next-key 锁(Next-key Lock)又称临键锁。
其中间隙锁只在RR隔离级别下生效,其目的是为了防止产生幻读。Next-key 锁(Next-key Lock)相当于一个索引记录锁加上该记录之前的一个间隙锁。
1、记录锁
记录锁(Record Lock)是针对索引记录(index record)的锁定。例如,SELECT * FROM t WHERE id = 1 FOR UPDATE;
会阻止其他事务对表 t 中 id = 1 的数据执行插入、更新,以及删除操作。
通过主键操作单个值
id 是表 t 的主键,我们先在 T1 中执行以下命令:
-- T1
mysql> SET GLOBAL innodb_status_output=ON;
Query OK, 0 rows affected (0.00 sec)
mysql> SET GLOBAL innodb_status_output_locks=ON;
Query OK, 0 rows affected (0.00 sec)
mysql> begin;
Query OK, 0 rows affected (0.00 sec)
mysql> SELECT * FROM t WHERE id = 1 FOR UPDATE;
+----+------+------+------+
| id | c1 | c2 | c3 |
+----+------+------+------+
| 1 | 1 | 1 | 1 |
+----+------+------+------+
1 row in set (0.00 sec)
全局变量 innodb_status_output 和 innodb_status_output_locks 用于控制 InnoDB 标准监控和锁监控,我们利用监控查看锁的使用情况。然后 T1 锁定了 id = 1 的记录,此时 T2 无法修改该记录:
-- T2
mysql> SELECT * FROM t WHERE id = 1 for update;
ERROR 1205 (HY000): Lock wait timeout exceeded; try restarting transaction
使用SHOW ENGINE INNODB STATUS
命令查看 InnoDB 监控中关于锁的事务数据,可以看到以下内容:
---TRANSACTION 43764, ACTIVE 4 sec
2 lock struct(s), heap size 1136, 1 row lock(s)
MySQL thread id 103, OS thread handle 140437513750272, query id 23734 localhost root
TABLE LOCK table `hrdb`.`t` trx id 43764 lock mode IX
RECORD LOCKS space id 101 page no 4 n bits 72 index PRIMARY of table `hrdb`.`t` trx id 43764 lock_mode X locks rec but not gap
Record lock, heap no 2 PHYSICAL RECORD: n_fields 6; compact format; info bits 0
0: len 4; hex 80000001; asc ;;
1: len 6; hex 00000000aaec; asc ;;
2: len 7; hex 820000008f0110; asc ;;
3: len 4; hex 80000001; asc ;;
4: len 4; hex 80000001; asc ;;
5: len 4; hex 80000001; asc ;;
日志显示存在 2 个锁结构,锁定了一个记录;表 t 上存在 IX 锁,主键索引上存在一个 X 记录锁,同时还显示了记录对应的数据值。注意but not gap
,下文我们会介绍间隙锁(Gap Lock)。最后在 T1 中释放锁:
-- T1
mysql> commit;
Query OK, 0 rows affected (0.00 sec)
通过唯一索引操作单个值
c1 字段上存在唯一索引,我们先在 T1 中执行以下命令:
-- T1
mysql> begin;
Query OK, 0 rows affected (0.00 sec)
mysql> SELECT * FROM t WHERE c1 = 1 FOR UPDATE;
+----+------+------+------+
| id | c1 | c2 | c3 |
+----+------+------+------+
| 1 | 1 | 1 | 1 |
+----+------+------+------+
1 row in set (0.00 sec)
使用SHOW ENGINE INNODB STATUS
命令查看 InnoDB 监控中关于锁的事务数据,可以看到以下内容:
---TRANSACTION 43761, ACTIVE 47 sec
3 lock struct(s), heap size 1136, 2 row lock(s)
MySQL thread id 103, OS thread handle 140437513750272, query id 23722 localhost root
TABLE LOCK table `hrdb`.`t` trx id 43761 lock mode IX
RECORD LOCKS space id 101 page no 5 n bits 72 index idx_t_c1 of table `hrdb`.`t` trx id 43761 lock_mode X locks rec but not gap
Record lock, heap no 2 PHYSICAL RECORD: n_fields 2; compact format; info bits 0
0: len 4; hex 80000001; asc ;;
1: len 4; hex 80000001; asc ;;
RECORD LOCKS space id 101 page no 4 n bits 72 index PRIMARY of table `hrdb`.`t` trx id 43761 lock_mode X locks rec but not gap
Record lock, heap no 2 PHYSICAL RECORD: n_fields 6; compact format; info bits 0
0: len 4; hex 80000001; asc ;;
1: len 6; hex 00000000aaec; asc ;;
2: len 7; hex 820000008f0110; asc ;;
3: len 4; hex 80000001; asc ;;
4: len 4; hex 80000001; asc ;;
5: len 4; hex 80000001; asc ;;
日志显示存在 3 个锁结构,锁定了 2 个记录;表 t 上存在 IX 锁,索引 idx_t_c1 上存在一个 X 记录锁,主键索引上存在一个 X 记录锁。最后在 T1 中释放锁:
-- T1
mysql> commit;
Query OK, 0 rows affected (0.00 sec)
记录锁永远都是锁定索引记录,锁定非聚集索引会先锁定聚集索引。如果表中没有定义索引,InnoDB 默认为表创建一个隐藏的聚簇索引,并且使用该索引锁定记录。
2、间隙锁
间隙锁(Gap Lock)锁定的是索引记录之间的间隙、第一个索引之前的间隙或者最后一个索引之后的间隙。例如,SELECT * FROM t WHERE c1 BETWEEN 1 and 10 FOR UPDATE;会阻止其他事务将 1 到 10 之间的任何值插入到 c1 字段中,即使该列不存在这样的数据;因为这些值都会被锁定。
加锁规则有以下特性,我们会在后面的案例中逐一解释:
- 1.加锁的基本单位是(next-key lock),他是前开后闭原则(这里实际是指的记录锁与间隙锁的组合)
- 2.插叙过程中访问的对象会增加锁
- 3.索引上的等值查询--给唯一索引加锁的时候,next-key lock升级为记录锁(注:在>=或between时只对主键生效,唯一但非主键索引不生效)。
- 4.索引上的等值查询--向右遍历时最后一个值不满足查询需求时,next-key lock 退化为间隙锁
- 5.唯一索引上的范围查询会访问到不满足条件的第一个值为止
- 使用普通索引锁定;在普通索引列上,不管是何种查询,只要加锁,都会产生间隙锁。
- 使用多列唯一索引会产生间隙锁。
- 使用唯一索引锁定多行记录(范围查找)。
- 使用唯一索引,对于指定查询某一条记录的加锁语句,如果该记录不存在,会产生记录锁和间隙锁,如果记录存在,则只会产生记录锁。
通过主键操作范围值
首先在 T1 中执行以下命令锁住数据范围:
-- T1 mysql> begin; Query OK, 0 rows affected (0.00 sec) mysql> SELECT * FROM t WHERE id BETWEEN 1 and 10 FOR UPDATE; +----+------+------+------+ | id | c1 | c2 | c3 | +----+------+------+------+ | 1 | 1 | 1 | 1 | | 2 | 2 | 3 | 4 | | 3 | 3 | 6 | 9 | +----+------+------+------+ 3 rows in set (0.00 sec)
表 t 中只有 3 条记录,id = 4 的记录不存在;即便如此,T2 仍然无法插入该记录:
-- T2 mysql> insert into t(c1,c2,c3) values (4,8,12); ERROR 1205 (HY000): Lock wait timeout exceeded; try restarting transaction
再次使用SHOW ENGINE INNODB STATUS
命令查看 InnoDB 监控中关于锁的事务数据,可以看到以下内容:
---TRANSACTION 43765, ACTIVE 4 sec 3 lock struct(s), heap size 1136, 4 row lock(s) MySQL thread id 103, OS thread handle 140437513750272, query id 23741 localhost root TABLE LOCK table `hrdb`.`t` trx id 43765 lock mode IX RECORD LOCKS space id 101 page no 4 n bits 72 index PRIMARY of table `hrdb`.`t` trx id 43765 lock_mode X locks rec but not gap Record lock, heap no 2 PHYSICAL RECORD: n_fields 6; compact format; info bits 0 0: len 4; hex 80000001; asc ;; 1: len 6; hex 00000000aaec; asc ;; 2: len 7; hex 820000008f0110; asc ;; 3: len 4; hex 80000001; asc ;; 4: len 4; hex 80000001; asc ;; 5: len 4; hex 80000001; asc ;; RECORD LOCKS space id 101 page no 4 n bits 72 index PRIMARY of table `hrdb`.`t` trx id 43765 lock_mode X Record lock, heap no 1 PHYSICAL RECORD: n_fields 1; compact format; info bits 0 0: len 8; hex 73757072656d756d; asc supremum;; Record lock, heap no 3 PHYSICAL RECORD: n_fields 6; compact format; info bits 0 0: len 4; hex 80000002; asc ;; 1: len 6; hex 00000000aaec; asc ;; 2: len 7; hex 820000008f011d; asc ;; 3: len 4; hex 80000002; asc ;; 4: len 4; hex 80000003; asc ;; 5: len 4; hex 80000004; asc ;; Record lock, heap no 4 PHYSICAL RECORD: n_fields 6; compact format; info bits 0 0: len 4; hex 80000003; asc ;; 1: len 6; hex 00000000aaec; asc ;; 2: len 7; hex 820000008f012a; asc *;; 3: len 4; hex 80000003; asc ;; 4: len 4; hex 80000006; asc ;; 5: len 4; hex 80000009; asc ;;
日志显示存在 3 个锁结构,锁定了 4 个索引记录;表 t 上存在 IX 锁,主键索引上存在 1 个 X 记录锁(id = 1)和 3 个间隙锁((1, 2]、(2, 3]、supremum);其中 supremum 代表了大于 3 的间隙((3, positive infinity))。实际上这里的间隙锁属于 Next-key 锁,相当于间隙锁加记录锁,下文将会介绍。
此时,我们可以插入 id 小于 1 的数据;但是不能插入 id 大于 10 的数据。
通过唯一索引操作范围值
首先在 T1 中执行以下命令锁住数据范围:
-- T1 mysql> begin; Query OK, 0 rows affected (0.00 sec) mysql> SELECT * FROM t WHERE c1 BETWEEN 1 and 10 FOR UPDATE; +----+------+------+------+ | id | c1 | c2 | c3 | +----+------+------+------+ | 1 | 1 | 1 | 1 | | 2 | 2 | 3 | 4 | | 3 | 3 | 6 | 9 | +----+------+------+------+ 3 rows in set (0.00 sec)
再次使用SHOW ENGINE INNODB STATUS
命令查看 InnoDB 监控中关于锁的事务数据,可以看到以下内容:
---TRANSACTION 43824, ACTIVE 153 sec 3 lock struct(s), heap size 1136, 7 row lock(s) MySQL thread id 103, OS thread handle 140437513750272, query id 23852 localhost root TABLE LOCK table `hrdb`.`t` trx id 43824 lock mode IX RECORD LOCKS space id 102 page no 5 n bits 72 index idx_t_c1 of table `hrdb`.`t` trx id 43824 lock_mode X Record lock, heap no 1 PHYSICAL RECORD: n_fields 1; compact format; info bits 0 0: len 8; hex 73757072656d756d; asc supremum;; Record lock, heap no 2 PHYSICAL RECORD: n_fields 2; compact format; info bits 0 0: len 4; hex 80000001; asc ;; 1: len 4; hex 80000001; asc ;; Record lock, heap no 3 PHYSICAL RECORD: n_fields 2; compact format; info bits 0 0: len 4; hex 80000002; asc ;; 1: len 4; hex 80000002; asc ;; Record lock, heap no 4 PHYSICAL RECORD: n_fields 2; compact format; info bits 0 0: len 4; hex 80000003; asc ;; 1: len 4; hex 80000003; asc ;; RECORD LOCKS space id 102 page no 4 n bits 72 index PRIMARY of table `hrdb`.`t` trx id 43824 lock_mode X locks rec but not gap Record lock, heap no 2 PHYSICAL RECORD: n_fields 6; compact format; info bits 0 0: len 4; hex 80000001; asc ;; 1: len 6; hex 00000000ab2b; asc +;; 2: len 7; hex 82000000a70110; asc ;; 3: len 4; hex 80000001; asc ;; 4: len 4; hex 80000001; asc ;; 5: len 4; hex 80000001; asc ;; Record lock, heap no 3 PHYSICAL RECORD: n_fields 6; compact format; info bits 0 0: len 4; hex 80000002; asc ;; 1: len 6; hex 00000000ab2b; asc +;; 2: len 7; hex 82000000a7011d; asc ;; 3: len 4; hex 80000002; asc ;; 4: len 4; hex 80000003; asc ;; 5: len 4; hex 80000004; asc ;; Record lock, heap no 4 PHYSICAL RECORD: n_fields 6; compact format; info bits 0 0: len 4; hex 80000003; asc ;; 1: len 6; hex 00000000ab2b; asc +;; 2: len 7; hex 82000000a7012a; asc *;; 3: len 4; hex 80000003; asc ;; 4: len 4; hex 80000006; asc ;; 5: len 4; hex 80000009; asc ;;
日志显示存在 3 个锁结构,锁定了 7 个索引记录;表 t 上存在 IX 锁,索引 idx_t_c1 上存在 4 个间隙锁((negative infinity, 1]、(1, 2]、(2, 3]、supremum),其中 supremum 代表了大于 3 的间隙((3, positive infinity));主键索引上存在 3 个 X 记录锁,锁定了 3 个主键值。实际上这里的间隙锁属于 Next-key 锁,相当于间隙锁加记录锁,下文将会介绍。
此时,我们无法插入任何数据。
需要注意的是,不同事务可以获取一个间隙上互相冲突的锁。例如,事务 A 在一个间隙上获取了共享的间隙锁(间隙 S 锁),事务 B 可以在同一间隙上获取排他的间隙锁(间隙 X 锁)。允许存在互相冲突的间隙锁的原因在于,如果从索引中清除某个记录,必须合并不同事务在记录上获取的间隙锁。
InnoDB 间隙锁的唯一目的是阻止其他事务在间隙中插入数据。间隙锁可以共存,一个事务的间隙锁不会阻止另一个事务在同一个间隙上获取间隙锁。共享间隙锁和排他间隙锁之间没有区别,彼此不冲突,它们的作用相同。
间隙锁可以显式禁用,例如将事务隔离级别设置为 READ COMMITTED。此时,查找和索引扫描不会使用间隙锁,间隙锁只用于外键约束和重复键的检查。
使用 READ COMMITTED 隔离级别还会带来其他影响,MySQL 在判断 WHERE 条件之后会释放不满足条件的数据行上的记录锁。对于 UPDATE 语句,InnoDB 执行“半一致”读取;以便将数据的最新版本返回给 MySQL,MySQL 就可以确定该行是否满足 UPDATE 中的 WHERE 条件。
3、Next-key 锁
Next-key 锁(Next-key Lock)相当于一个索引记录锁加上该记录之前的一个间隙锁。
InnoDB 实现行级锁的方式如下:当搜索或扫描表索引时,在遇到的索引记录上设置共享锁或排它锁。因此,InnoDB 行级锁实际上是索引记录锁。
假设一个索引中包含数据 10、11、13 和 20。该索引中可能的 next-key 锁包含以下范围,其中圆括号表示排除端点值,方括号表示包含端点值:
(negative infinity, 10] (10, 11] (11, 13] (13, 20] (20, positive infinity) -- 显示为 supermum
对于最后一个间隔,next-key 锁将会锁定最大索引值(20)之后的间隙;伪记录“supermum”的值大于索引中任何值,它不是真正的索引记录。(10, 11) 是一个间隙锁的锁定范围,(10, 11] 是一个 next-key 锁的锁定范围。
默认隔离级别(REPEATABLE READ )下,InnoDB 通过 next-key 锁进行查找和索引扫描,用于防止幻读;因为它会锁定范围值,不会导致两次查询结果的数量不同。
通过普通索引操作单个值
c2 字段上存在非唯一索引,我们先在 T1 中执行以下命令:
-- T1 mysql> begin; Query OK, 0 rows affected (0.00 sec) mysql> SELECT * FROM t WHERE c2 = 1 FOR UPDATE; +----+------+------+------+ | id | c1 | c2 | c3 | +----+------+------+------+ | 1 | 1 | 1 | 1 | +----+------+------+------+ 1 row in set (0.00 sec)
使用SHOW ENGINE INNODB STATUS
命令查看 InnoDB 监控中关于锁的事务数据,可以看到以下内容:
---TRANSACTION 43830, ACTIVE 6 sec 4 lock struct(s), heap size 1136, 3 row lock(s) MySQL thread id 103, OS thread handle 140437513750272, query id 23871 localhost root TABLE LOCK table `hrdb`.`t` trx id 43830 lock mode IX RECORD LOCKS space id 102 page no 6 n bits 72 index idx_t_c2 of table `hrdb`.`t` trx id 43830 lock_mode X Record lock, heap no 2 PHYSICAL RECORD: n_fields 2; compact format; info bits 0 0: len 4; hex 80000001; asc ;; 1: len 4; hex 80000001; asc ;; RECORD LOCKS space id 102 page no 4 n bits 72 index PRIMARY of table `hrdb`.`t` trx id 43830 lock_mode X locks rec but not gap Record lock, heap no 2 PHYSICAL RECORD: n_fields 6; compact format; info bits 0 0: len 4; hex 80000001; asc ;; 1: len 6; hex 00000000ab2b; asc +;; 2: len 7; hex 82000000a70110; asc ;; 3: len 4; hex 80000001; asc ;; 4: len 4; hex 80000001; asc ;; 5: len 4; hex 80000001; asc ;; RECORD LOCKS space id 102 page no 6 n bits 72 index idx_t_c2 of table `hrdb`.`t` trx id 43830 lock_mode X locks gap before rec Record lock, heap no 3 PHYSICAL RECORD: n_fields 2; compact format; info bits 0 0: len 4; hex 80000003; asc ;; 1: len 4; hex 80000002; asc ;;
日志显示存在 4 个锁结构,锁定了 3 个索引项;表 t 上存在 IX 锁,索引 idx_t_c2 上存在一个 next-key 锁(c2 = 1,锁定了 (negative infinity, 1])和一个 X 间隙锁((1, 3)),主键索引上存在一个 X 记录锁(id = 1)。
此时其他事务无法在 c2 中插入小于 3 的值,但是可以插入大于等于 3 的值。最后在 T1 中释放锁:
-- T1 mysql> commit; Query OK, 0 rows affected (0.00 sec)
如果索引有唯一属性,则 InnnoDB 会自动将 next-key 锁降级为记录锁。我们在前面已经给出了记录锁的示例。
通过普通索引操作范围值
如果利用 c2 字段作为条件操作范围值,加锁情况与通过唯一索引(c1)操作范围值相同。可以参考上文示例。
无索引操作单个值或范围值
c3 字段上没有索引,我们先在 T1 中执行以下命令:
-- T1 mysql> begin; Query OK, 0 rows affected (0.00 sec) mysql> SELECT * FROM t WHERE c3 = 1 FOR UPDATE; +----+------+------+------+ | id | c1 | c2 | c3 | +----+------+------+------+ | 1 | 1 | 1 | 1 | +----+------+------+------+ 1 row in set (0.00 sec)
使用SHOW ENGINE INNODB STATUS
命令查看 InnoDB 监控中关于锁的事务数据,可以看到以下内容:
---TRANSACTION 43848, ACTIVE 5 sec 2 lock struct(s), heap size 1136, 4 row lock(s) MySQL thread id 103, OS thread handle 140437513750272, query id 23917 localhost root TABLE LOCK table `hrdb`.`t` trx id 43848 lock mode IX RECORD LOCKS space id 102 page no 4 n bits 80 index PRIMARY of table `hrdb`.`t` trx id 43848 lock_mode X Record lock, heap no 1 PHYSICAL RECORD: n_fields 1; compact format; info bits 0 0: len 8; hex 73757072656d756d; asc supremum;; Record lock, heap no 2 PHYSICAL RECORD: n_fields 6; compact format; info bits 0 0: len 4; hex 80000001; asc ;; 1: len 6; hex 00000000ab2b; asc +;; 2: len 7; hex 82000000a70110; asc ;; 3: len 4; hex 80000001; asc ;; 4: len 4; hex 80000001; asc ;; 5: len 4; hex 80000001; asc ;; Record lock, heap no 3 PHYSICAL RECORD: n_fields 6; compact format; info bits 0 0: len 4; hex 80000002; asc ;; 1: len 6; hex 00000000ab2b; asc +;; 2: len 7; hex 82000000a7011d; asc ;; 3: len 4; hex 80000002; asc ;; 4: len 4; hex 80000003; asc ;; 5: len 4; hex 80000004; asc ;; Record lock, heap no 4 PHYSICAL RECORD: n_fields 6; compact format; info bits 0 0: len 4; hex 80000003; asc ;; 1: len 6; hex 00000000ab2b; asc +;; 2: len 7; hex 82000000a7012a; asc *;; 3: len 4; hex 80000003; asc ;; 4: len 4; hex 80000006; asc ;; 5: len 4; hex 80000009; asc ;;
日志显示存在 2 个锁结构,锁定了 4 个索引项;表 t 上存在 IX 锁,主键索引上存在 4 个 next-key 锁,锁定了所有的主键范围。此时其他事务无法插入任何数据。
最后在 T1 中释放锁:
-- T1 mysql> commit; Query OK, 0 rows affected (0.00 sec)
如果我们将语句修改为SELECT * FROM t WHERE c3 between 1 and 10 FOR UPDATE;
,通过无索引的字段操作范围值,也会锁定主键的所有范围。这也就是为什么 MySQL 推荐通过索引操作数据,最好是主键。
4、插入意向锁
插入意向锁(Insert Intention Lock)是在插入数据行之前,由 INSERT 操作设置的一种间隙锁。插入意向锁表示一种插入的意图,如果插入到相同间隙中的多个事务没有插入相同位置,则不需要互相等待。假设存在索引记录 4 和 7。两个事务分别尝试插入 5 和 6,它们在获取行排他锁之前,分别使用插入意向锁来锁定 4 到 7 之间的间隙;但是不会相互阻塞,因为插入的是不同的行。
注:插入意向锁并非意向锁,而是一种特殊的间隙锁。
在 T1 中对 c2 大于 3 的索引记录设置排它锁,这个排它锁包含记录 6 之前的间隙锁:
-- T1 mysql> begin; Query OK, 0 rows affected (0.00 sec) mysql> SELECT * FROM t WHERE c2 > 3 FOR UPDATE; +----+------+------+------+ | id | c1 | c2 | c3 | +----+------+------+------+ | 3 | 3 | 6 | 9 | +----+------+------+------+ 1 row in set (0.00 sec)
使用 SHOW ENGINE INNODB STATUS
命令可以显示锁事务数据:
---TRANSACTION 43853, ACTIVE 38 sec 3 lock struct(s), heap size 1136, 3 row lock(s) MySQL thread id 103, OS thread handle 140437513750272, query id 23931 localhost root TABLE LOCK table `hrdb`.`t` trx id 43850 lock mode IX RECORD LOCKS space id 102 page no 6 n bits 72 index idx_t_c2 of table `hrdb`.`t` trx id 43850 lock_mode X Record lock, heap no 1 PHYSICAL RECORD: n_fields 1; compact format; info bits 0 0: len 8; hex 73757072656d756d; asc supremum;; Record lock, heap no 4 PHYSICAL RECORD: n_fields 2; compact format; info bits 0 0: len 4; hex 80000006; asc ;; 1: len 4; hex 80000003; asc ;; RECORD LOCKS space id 102 page no 4 n bits 80 index PRIMARY of table `hrdb`.`t` trx id 43850 lock_mode X locks rec but not gap Record lock, heap no 4 PHYSICAL RECORD: n_fields 6; compact format; info bits 0 0: len 4; hex 80000003; asc ;; 1: len 6; hex 00000000ab2b; asc +;; 2: len 7; hex 82000000a7012a; asc *;; 3: len 4; hex 80000003; asc ;; 4: len 4; hex 80000006; asc ;; 5: len 4; hex 80000009; asc ;;
在 T2 中将 c2 = 4 插入间隙中,该事务在等待获取独占锁时使用插入意向锁:
-- T2 mysql> insert into t(c1,c2,c3) values (4,4,4);
使用 SHOW ENGINE INNODB STATUS
命令可以显示插入意向锁事务数据:
---TRANSACTION 43854, ACTIVE 5 sec inserting mysql tables in use 1, locked 1 LOCK WAIT 2 lock struct(s), heap size 1136, 1 row lock(s), undo log entries 1 MySQL thread id 106, OS thread handle 140437512111872, query id 23957 localhost root update insert into t(c1,c2,c3) values (4,4,4) ------- TRX HAS BEEN WAITING 5 SEC FOR THIS LOCK TO BE GRANTED: RECORD LOCKS space id 102 page no 6 n bits 72 index idx_t_c2 of table `hrdb`.`t` trx id 43854 lock_mode X locks gap before rec insert intention waiting Record lock, heap no 4 PHYSICAL RECORD: n_fields 2; compact format; info bits 0 0: len 4; hex 80000006; asc ;; 1: len 4; hex 80000003; asc ;; ------------------ TABLE LOCK table `hrdb`.`t` trx id 43854 lock mode IX RECORD LOCKS space id 102 page no 6 n bits 72 index idx_t_c2 of table `hrdb`.`t` trx id 43854 lock_mode X locks gap before rec insert intention waiting Record lock, heap no 4 PHYSICAL RECORD: n_fields 2; compact format; info bits 0 0: len 4; hex 80000006; asc ;; 1: len 4; hex 80000003; asc ;;
其中,locks gap before rec insert intention 表示插入意向锁。由于 T1 锁定了 3 和 6 之间的范围,T2 需要等待;如果 T1 插入数据 4,T2 插入数据 5,互相之间不需要等待。
插入意向锁的作用是为了提高并发插入的性能。间隙锁不允许多个事务同时插入同一个索引间隙,但是插入意向锁允许多个事务同时插入同一个索引间隙内的不同数据值。
四、案例分析
一下案例均在RR隔离级别下,关闭事务自动提交
SELECT @@tx_isolation
SET autocommit = 0; SHOW VARIABLES LIKE 'autocommit';
1、主键与唯一索引及普通索引的范围查询
DROP TABLE IF EXISTS `t`; CREATE TABLE `t` ( `id` int(11) NOT NULL AUTO_INCREMENT, `c1` int(11) NULL DEFAULT NULL, `c2` int(11) NULL DEFAULT NULL, `c3` int(11) NULL DEFAULT NULL, PRIMARY KEY (`id`) USING BTREE, UNIQUE INDEX `idx_t_c1`(`c1`) USING BTREE, INDEX `idx_t_c2`(`c2`) USING BTREE ) ENGINE = InnoDB AUTO_INCREMENT = 9 CHARACTER SET = utf8mb4 COLLATE = utf8mb4_general_ci ROW_FORMAT = Dynamic; INSERT INTO `t` VALUES (1, 1, 1, 1); INSERT INTO `t` VALUES (3, 3, 3, 3); INSERT INTO `t` VALUES (4, 4, 4, 4); INSERT INTO `t` VALUES (5, 5, 5, 5); INSERT INTO `t` VALUES (6, 6, 6, 6); INSERT INTO `t` VALUES (7, 7, 7, 7); INSERT INTO `t` VALUES (10, 10, 10, 10);
实测证实(图省事多次验证后下面写在了一起,如需实测,可执行sql自行执行):
事务一
select * from t where id >3 and id < 7 for update; select * from t where id >=3 and id <= 7 for update; select * from t where c1 >=3 and c1<= 7 for update; select * from t where c1 >3 and c1< 7 for update;
事务二
SET autocommit = 0; INSERT INTO `performance_rpmt`.`t`(`id`, `c1`, `c2`, `c3`) VALUES (2, 2, 2, 2); INSERT INTO `performance_rpmt`.`t`(`id`, `c1`, `c2`, `c3`) VALUES (9, 9, 9, 9); commit;
直接上结论:
- 主键id 插入(2, 2, 2, 2)时,
- id > 3 或 id >= 3,都不会锁;
- id >3 时, 事务二如果执行 select * from t where id =3 for update; 不会被锁,所以第一个间隙锁区间是(3,4]临键锁。
- 主键id 插入(9, 9, 9, 9)时,
- id < 7 不会锁; id <= 7 会锁;
- 但id < 7时, 事务二如果执行 select * from t where id =7 for update; 会被锁,所以最后的间隙锁区间是(6,7]临键锁。
- 唯一或普通索引 c1或c2 插入(2, 2, 2, 2)时,
- >3 或 >= 3 都会锁;
- c1>3 时, 事务二如果执行 select * from t where c1 =3 for update; 会被锁,所以第一个间隙锁区间是(1,3]临键锁。
- 唯一或普通索引 c1或c2 插入(9, 9, 9, 9)时,
- <7 或<= 7 都会锁;
- <7时,事务二如果执行 select * from t where c1=7 for update; 会被锁,所以最后的间隙锁区间是(6,7]临键锁。
所以经以上实测,也论证了加锁特性的第一条与第四条。
2、案例二
案例数据
id(主键) | c(普通索引) | d(无索引) |
---|---|---|
5 | 5 | 5 |
10 | 10 | 10 |
15 | 15 | 15 |
20 | 20 | 20 |
25 | 25 | 25 |
以上数据为了解决幻读问题,更新的时候不只是对上述的五条数据增加行锁,还对于中间的取值范围增加了6间隙锁,(-∞,5](5,10](10,15](15,20](20,25](25,+supernum] (其中supernum是数据库维护的最大的值。为了保证间隙锁都是左开右闭原则。)
案例一:间隙锁简单案例
步骤 | 事务A | 事务B |
---|---|---|
1 | begin; select * from t where id = 11 for update; |
- |
2 | - | insert into user value(12,12,12) |
3 | commit; | - |
当有如下事务A和事务B时,事务A会对数据库表增加(10,15]这个区间锁,这时insert id = 12 的数据的时候就会因为区间锁(10,15]而被锁住无法执行。
案例二: 间隙锁死锁问题
步骤 | 事务A | 事务B |
---|---|---|
1 | begin; select * from t where id = 9 for update; |
- |
2 | - | begin; select * from t where id = 6 for update; |
3 | - | insert into user value(7,7,7) |
4 | insert into user value(7,7,7) |
- |
不同于写锁相互之间是互斥的原则,间隙锁之间不是互斥的,如果一个事务A获取到了(5,10]之间的间隙锁,另一个事务B也可以获取到(5,10]之间的间隙锁。这时就可能会发生死锁问题,如下案例。
事务A获取到(5,10]之间的间隙锁不允许其他的DDL操作,在事务提交,间隙锁释放之前,事务B也获取到了间隙锁(5,10],这时两个事务就处于死锁状态。
案例三: 等值查询—主键(如果是唯一索引这里效果是一致的)
步骤 | 事务A | 事务B | 事务C |
---|---|---|---|
1 | begin; update u set d= d+ 1 where id = 7; |
- | - |
2 | - | insert into u (8,8,8); |
- |
4 | - | - | update set d = d+ 1 where id = 10 |
1.加锁的范围是(5,10]的范围锁
2.由于数据是等值查询,并且表中最后数据id = 10 不满足id= 7的查询要求,故id=10 的行级锁退化为间隙锁,(5,10)
3.所以事务B中id=8会被锁住,而id=10的时候不会被锁住。
案例四: 等值查询—普通索引(比较特殊的一种情况)
步骤 | 事务A | 事务B | 事务C |
---|---|---|---|
1 | begin; select id form t where c = 5 lock in share mode; |
- | - |
2 | - | update t set d = d + 1 where id = 5 | - |
4 | - | - | insert into values (7,7,7) |
1.加锁的范围是(0,5],(5,10]的范围锁
2.由于c是普通索引,根据原则4,搜索到5后继续向后遍历直到搜索到10才放弃,故加锁范围为(5,10]
3.由于查询是等值查询,并且最后一个值不满足查询要求,故间隙锁退化为(5,10)
4.因为加锁是对普通索引c加锁,而且因为索引覆盖,没有对主键进行加锁,所以事务B执行正常
5.因为加锁范围(5,10)故事务C执行阻塞
6.需要注意的是,lock in share mode 因为覆盖索引故没有锁主键索引,如果使用for update 程序会觉得之后会执行更新操作故会将主键索引一同锁住。
案例五: 范围查询—主键
步骤 | 事务A | 事务B | 事务C |
---|---|---|---|
1 | begin; select * form t where id >= 10 and id <11 for update |
- | - |
2 | - | insert into values(8,8,8) insert into values(13,13,13) |
- |
4 | - | - | update t set d = d+ 1 where id = 15 |
- 因为等值查询有id =10 根据原则3间隙锁升级为行锁,故剩余锁[10,15]
- 因为查询并不是等值查询,故[10,15]不会退化成[10,15)
- 故事务B(13,13,13)阻塞,事务C阻塞。
案例六: 普通索引-等值问题
上面的数据增加一行(30,10,30),这样在数据库中存在的c=10的就有两条记录
步骤 | 事务A | 事务B | 事务C |
---|---|---|---|
1 | begin; delete from t where c = 10 |
- | - |
2 | - | insert into values(12,12,12) |
- |
4 | - | - | update t set d = d+ 1 where c = 15 |
- next-key lock 增加范围锁(5,10],(10,15]
-
因为是等值查询故退化为(5,10],(10,15),故事务B阻塞,事务C执行成功
3、案例三
准备数据:
CREATE TABLE `test1` ( `id` int(1) NOT NULL AUTO_INCREMENT, `number` int(1) NOT NULL COMMENT '数字', PRIMARY KEY (`id`), KEY `number` (`number`) USING BTREE ) ENGINE=InnoDB AUTO_INCREMENT=1 DEFAULT CHARSET=utf8;
INSERT INTO `test1` VALUES (1, 1); INSERT INTO `test1` VALUES (5, 3); INSERT INTO `test1` VALUES (7, 8); INSERT INTO `test1` VALUES (11, 12);
在进行测试之前,我们先来看看test1表中 number 索引存在的隐藏间隙:
- (-infinity, 1]
- (1, 3]
- (3, 8]
- (8, 12]
- (12, +infinity]
/* 开启事务1 */ BEGIN; /* 查询 number = 5 的数据并加记录锁 */ SELECT * FROM `test1` WHERE `number` = 3 FOR UPDATE; # 注意:以下的语句不是放在一个事务中执行,而是分开多次执行,每次事务中只有一条添加语句 /* 事务2插入一条 number = 0 的数据 */ INSERT INTO `test1` (`number`) VALUES (0); # 正常执行 /* 事务3插入一条 number = 1 的数据 */ INSERT INTO `test1` (`number`) VALUES (1); # 被阻塞 /* 事务4插入一条 number = 2 的数据 */ INSERT INTO `test1` (`number`) VALUES (2); # 被阻塞 /* 事务5插入一条 number = 4 的数据 */ INSERT INTO `test1` (`number`) VALUES (4); # 被阻塞 /* 事务6插入一条 number = 8 的数据 */ INSERT INTO `test1` (`number`) VALUES (8); # 正常执行 /* 事务7插入一条 number = 9 的数据 */ INSERT INTO `test1` (`number`) VALUES (9); # 正常执行 /* 事务8插入一条 number = 10 的数据 */ INSERT INTO `test1` (`number`) VALUES (10); # 正常执行 /* 提交事务1 */ COMMIT;
我们会发现有些语句可以正常执行,有些语句被阻塞了。我们再来看看我们表中的数据:
这里可以看到,number (1 - 8) 的间隙中,插入语句都被阻塞了,而不在这个范围内的语句,正常执行,这就是因为有间隙锁的原因。我们再进行以下的测试,方便我们更好的理解间隙锁的区域(我们要将数据还原成原来的那样):
/* 开启事务1 */ BEGIN; SELECT * FROM `test1` WHERE `number` = 3 FOR UPDATE; /* 事务1插入一条 id = 2, number = 1 的数据 */ INSERT INTO `test1` (`id`, `number`) VALUES (2, 1); # 阻塞 /* 事务2插入一条 id = 3, number = 2 的数据 */ INSERT INTO `test1` (`id`, `number`) VALUES (3, 2); # 阻塞 /* 事务3插入一条 id = 6, number = 8 的数据 */ INSERT INTO `test1` (`id`, `number`) VALUES (6, 8); # 阻塞 /* 事务4插入一条 id = 8, number = 8 的数据 */ INSERT INTO `test1` (`id`, `number`) VALUES (8, 8); # 正常执行 /* 事务5插入一条 id = 9, number = 9 的数据 */ INSERT INTO `test1` (`id`, `number`) VALUES (9, 9); # 正常执行 /* 事务6插入一条 id = 10, number = 12 的数据 */ INSERT INTO `test1` (`id`, `number`) VALUES (10, 12); # 正常执行 /* 事务7修改 id = 11, number = 12 的数据 */ UPDATE `test1` SET `number` = 5 WHERE `id` = 11 AND `number` = 12; # 阻塞 /* 提交事务1 */ COMMIT;
结果:
这里有一个奇怪的现象:
- 事务3添加 id = 6,number = 8 的数据,给阻塞了;
- 事务4添加 id = 8,number = 8 的数据,正常执行了。
- 事务7将 id = 11,number = 12 的数据修改为 id = 11, number = 5的操作,给阻塞了;
这是为什么呢?我们来看看下边的图,大家就明白了。
从图中可以看出,当 number 相同时,会根据主键 id 来排序,所以:
- 事务3添加的 id = 6,number = 8,这条数据是在 (3, 8) 的区间里边,所以会被阻塞;
- 事务4添加的 id = 8,number = 8,这条数据则是在(8, 12)区间里边,所以不会被阻塞;
- 事务7的修改语句相当于在 (3, 8) 的区间里边插入一条数据,所以也被阻塞了。
结论
- 在普通索引列上,不管是何种查询,只要加锁,都会产生间隙锁,这跟唯一索引不一样;
- 在普通索引跟唯一索引中,数据间隙的分析,数据行是优先根据普通索引排序,再根据唯一索引排序。
参考文章:
https://tonydong.blog.csdn.net/article/details/103324323
https://www.jianshu.com/p/32904ee07e56
https://zhuanlan.zhihu.com/p/48269420