MySQL 是怎么加行级锁的?为什么一会是 next-key 锁,一会是间隙锁,一会又是记录锁?( 六 )


MySQL 是怎么加行级锁的?为什么一会是 next-key 锁,一会是间隙锁,一会又是记录锁?

文章插图
  • 在 id = 1 这条记录的主键索引上,加了范围为 (-∞, 1] 的 next-key 锁 。意味着其他事务即无法更新或者删除 id = 1 的这一条记录,同时也无法插入 id 小于 1 的这一些新记录 。
  • 在 id = 5 这条记录的主键索引上,加了范围为 (1, 5] 的 next-key 锁 。意味着其他事务即无法更新或者删除 id = 5 的这一条记录 , 同时也无法插入 id 值为 2、3、4 的这一些新记录 。
我们也可以通过 select * from performance_schema.data_locks\G; 这条语句来看看事务 A 加了什么锁 。
输出结果如下,我这里只截取了行级锁的内容 。
MySQL 是怎么加行级锁的?为什么一会是 next-key 锁,一会是间隙锁,一会又是记录锁?

文章插图
从上图中的分析中,可以得到事务 A 在主键索引上加了两个 X 型 next-key 锁,分别是:
  • 在 id = 1 这条记录的主键索引上,加了范围为 (-∞, 1] 的 next-key 锁;
  • 在 id = 5 这条记录的主键索引上,加了范围为(1, 5 ] 的 next-key 锁 。
实验三:再来看针对「小于」的范围查询时,查询条件值的记录「存在」表中的情况 。
如果事务 A 的查询语句是小于的范围查询,且查询条件值的记录(id 为 5)存在于表中 。
select * from user where id < 5 for update;事务 A 加锁变化过程如下:
  1. 最开始要找的第一行是 id = 1,于是对该记录加的是范围为 (-∞, 1] 的 next-key 锁;
  2. 由于是范围查找,就会继续往后找存在的记录 , 扫描到的第二行是 id = 5,该记录是第一条不满足 id < 5 条件的记录,于是**该记录的锁会退化为间隙锁,锁范围是 (1,5)** 。
  3. 由于找到了第一条不满足 id < 5 条件的记录,于是停止扫描 。
可以得知 , 此时事务 A 在主键索引上加了两种 X 型锁:
![](https://cdn.xiaolincoding.com/gh/xiaolincoder/mysql/行级锁/唯一索引范围查询小于.drawio (1).png)
  • 在 id = 1 这条记录的主键索引上,加了范围为 (-∞, 1] 的 next-key 锁,意味着其他事务即无法更新或者删除 id = 1 的这一条记录 , 同时也无法插入 id 小于 1 的这一些新记录 。
  • 在 id = 5 这条记录的主键索引上,加了范围为 (1,5) 的间隙锁,意味着其他事务无法插入 id 值为 2、3、4 的这一些新记录 。
我们也可以通过 select * from performance_schema.data_locks\G; 这条语句来看看事务 A 加了什么锁 。
输出结果如下,我这里只截取了行级锁的内容 。
MySQL 是怎么加行级锁的?为什么一会是 next-key 锁,一会是间隙锁,一会又是记录锁?

文章插图
从上图中的分析中,可以得到事务 A 在主键索引上加了 X 型的范围为 (-∞, 1] 的 next-key 锁 , 和 X 型的范围为 (1, 5) 的间隙锁 。
因此,通过前面这三个实验,可以得知 。
在针对「小于或者小于等于」的唯一索引(主键索引)范围查询时,存在这两种情况会将索引的 next-key 锁会退化成间隙锁的:
  • 当条件值的记录「不在」表中时,那么不管是「小于」还是「小于等于」条件的范围查询,扫描到终止范围查询的记录时 , 该记录的主键索引中的 next-key 锁会退化成间隙锁,其他扫描到的记录,都是在这些记录的主键索引上加 next-key 锁 。
  • 当条件值的记录「在」表中时:
    • 如果是「小于」条件的范围查询,扫描到终止范围查询的记录时 , 该记录的主键索引中的 next-key 锁会退化成间隙锁,其他扫描到的记录,都是在这些记录的主键索引上,加 next-key 锁 。
    • 如果是「小于等于」条件的范围查询,扫描到终止范围查询的记录时,该记录的主键索引中的 next-key 锁「不会」退化成间隙锁,其他扫描到的记录,都是在这些记录的主键索引上加 next-key 锁 。
非唯一索引等值查询当我们用非唯一索引进行等值查询的时候,因为存在两个索引 , 一个是主键索引,一个是非唯一索引(二级索引) , 所以在加锁时,同时会对这两个索引都加锁,但是对主键索引加锁的时候,只有满足查询条件的记录才会对它们的主键索引加锁 。
针对非唯一索引等值查询时 , 查询的记录存不存在,加锁的规则也会不同: