MySQL语句加锁分析


    一条语句需要加的锁受到很多条件制约,比方说:

  • 事务的隔离级别
  • 语句执行时使用的索引(比如聚簇索引、唯一二级索引、普通二级索引)
  • 查询条件(比方说=、=<、>=等等)
  • 具体执行的语句类型

    🌹加锁只是解决并发事务执行过程中引起的脏写、脏读、不可重复读、幻读这些问题。🌹    不同情景下要解决的问题不一样,才导致加的锁不一样,千万不要为了加锁而加锁。

    这里把语句分为3种大类:
(1)普通的 SELECT 语句
(2)锁定读的语句
(3)INSERT 语句

1.普通的SELECT语句

    在读未提交 ❎隔离级别下,如果不加锁,是直接读取记录的最新版本,可能发生脏读、不可重复读和幻读问题。
    在 读提交 ✅隔离级别下,如果不加锁,在每次执行普通的 SELECT 语句时都会生成一个 ReadView ,这样解决了脏读问题,但没有解决不可重复读和幻读问题。
    在 可重复读👯‍♀️ 隔离级别下,不加锁,只在第一次执行普通的 SELECT 语句时生成一个 ReadView ,这样把脏读、不可重复读和幻读问题都解决了。
    在 串行读 ♒隔离级别下,需要分为两种情况讨论:
(1)在系统变量 autocommit=0 时,也就是 禁用自动提交 时,普通的SELECT 语句会被转为 SELECT … LOCK IN SHARE MODE 语句,也就是在读取记录前需要先获得记录的 S锁,具体的加锁情况和重复读隔离级别下一样,我们后边再分析。
(2)在系统变量 autocommit=1 时,也就是 启用自动提交 时,普通的SELECT 语句并不加锁,只是利用MVCC来生成一个ReadView去读取记录。
    为啥不加锁呢?因为启用自动提交意味着一个事务中只包含一条语句,一条语句也就没有啥不可重复读、幻读这样的问题了。

2.锁定读的语句

    我们把下边四种语句放到一起讨论:
语句一:SELECT … LOCK IN SHARE MODE;
语句二:SELECT … FOR UPDATE;
语句三:UPDATE …
语句四:DELETE …

    语句一 和 语句二 是 MySQL 中规定的两种锁定读的语法格式,而语句三和语句四由于在执行过程 需要 首先定位到被改动的记录 并给记录加锁,也可以被认为是一种锁定读。

读未提交❎、读提交✅ 隔离级别下

    值得注意的是,采用加锁方式解决并发事务带来的问题时,其实脏读和不可重复读在任何一个隔离级别下都不会发生(因为读-写操作需要排队进行)。

2.1 对于使用主键进行等值查询的情况

(1)
    使用👉 SELECT … LOCK IN SHARE MODE 👈为记录加锁,加的是 S型正经锁。如:

SELECT * FROM hero WHERE number = 8 LOCK IN SHARE MODE;

    这个语句执行时只需要访问一下聚簇索引中 number值为 8 的记录,所以只需要给它加一个 S型正经记录锁 就好了。

(2)
    使用 👉 SELECT … FOR UPDATE 👈为记录加锁,比方说:

SELECT * FROM hero WHERE number = 8 FOR UPDATE;

    这个语句执行时只需要访问一下聚簇索引中 number值为 8 的记录,所以只需要给它加一个 X型正经记录锁 就好了,如图所示:

(3)
①    使用👉 UPDATE👈 … 来为记录加锁,比方说:

UPDATE hero SET country = '汉' WHERE number = 8;

    这条 UPDATE 语句并没有更新二级索引列,加锁方式和上边所说的 SELECT … FOR UPDATE 语句一致,也就是 X型正经记录锁

②    如果👉 UPDATE 👈语句中更新了二级索引列,比方说:

UPDATE hero SET name = 'cao曹操' WHERE number = 8;

X型正经记录锁该语句的实际执行步骤是首先更新对应的 number 值为 8 的聚簇索引记录,再更新对应的二级索引记录,所以加锁的步骤就是:

  • 为 number 值为 8 的聚簇索引记录加上 X型正经记录锁(该记录对应的)。
  • 为该聚簇索引记录对应的 idx_name
    二级索引记录(也就是name值为’c曹操’,number值为8的那条二级索引记录)加上 X型正经记录锁。

(4)
    使用👉 DELETE 👈… 来为记录加锁,比方说:

DELETE FROM hero WHERE number = 8;

    我们平时所说的 “DELETE表中的一条记录” 其实意味着对 聚簇索引 和 所有的二级索引中对应的记录做 DELETE 操作,本例子中就是要先把 number 值为 8 的聚簇索引记录执行 DELETE 操作,然后把对应的 idx_name 二级索引记录删除,所以加锁的步骤和上边更新带有二级索引列的UPDATE语句一致。

2.2 对于使用主键进行范围查询的情况

(1)
    使用 👉SELECT … LOCK IN SHARE MODE 👈来为记录加锁,比方说:

SELECT * FROM hero WHERE number <= 8 LOCK IN SHARE MODE;

    这个语句看起来十分简单,但它的执行过程还是有一丢丢小复杂的:

  • 先到聚簇索引中定位到满足number <= 8的第一条记录,也就是number值为1的记录,然后为其加锁。
  • 判断一下该记录是否符合索引条件下推中的条件。
        前边介绍过一个称之为索引条件下推( Index Condition Pushdown,简称ICP)的功能,也就是把 查询中 与 被使用索引有关 的 查询条件 下推到 存储引擎 中判断,而不是返回到 server 层再判断。不过需要注意的是,索引条件下推只是为了减少回表次数,也就是减少读取完整的聚簇索引记录的次数,从而减少 IO操作。而对于聚簇索引而言不需要回表,它本身就包含着全部的列,也起不到减少 IO操作 的作用,所以规定这个==索引条件下推特性只适用于二级索引。==也就是说在本例中与被使用索引有关的条件是:number <= 8,而number列又是聚簇索引列,所以本例中并没有符合索引条件下推的查询条件,自然也就不需要判断该记录是否符合索引条件下推中的条件。
  • 判断一下该记录是否符合范围查询的边界条件
        因为在本例中是利用主键 number 进行范围查询,规定 每从 聚簇索引中取出一条记录时 都要判断一下该记录是否符合范围查询的边界条件,也就是number <= 8这个条件。如果符合的话将其返回给server层 继续处理,否则的话需要释放掉在该记录上加的锁,并给 server 层返回一个查询完毕的信息。
        对于number值为1的记录是符合这个条件的,所以会将其返回到server层继续处理。
  • 将该记录返回到server层继续判断。
        server层 如果收到存储引擎层提供的查询完毕的信息,就结束查询,否则继续判断那些没有进行索引条件下推的条件,在本例中就是继续判断 number <= 8 这个条件是否成立。噫,不是在第3步中已经判断过了么,怎么在这又判断一回?就是这么简单粗暴,把凡是没有经过索引条件下推的条件都需要放到server层再判断一遍。如果该记录符合剩余的条件(没有进行索引条件下推的条件),那么就把它发送给客户端,不然的话需要释放掉在该记录上加的锁。
        然后刚刚查询得到的这条记录(也就是 number 值为 1 的记录)组成的单向链表继续向后查找,得到了 number 值为 3 的记录,然后重复第2,3,4、5这几个步骤。
        但是这个过程有个问题,就是当找到number值为8的那条记录的时候,还得向后找一条记录(也就是 number 值为 15 的记录),在存储引擎读取这条记录的时候,也就是上述的第1步中,就得为这条记录加锁,然后在 第3步 时,判断该记录不符合 number <= 8 这个条件,又要释放掉这条记录的锁,这个过程导致 number 值为 15 的记录先被加锁,然后把锁释放掉。
        这个过程有意思的一点就是,如果你先在事务T1中执行:
# 事务T1
BEGIN;
SELECT * FROM hero WHERE number <= 8 LOCK IN SHARE MODE;
然后再到事务T2中执行:

# 事务T2
BEGIN;
SELECT * FROM hero WHERE number = 15 FOR UPDATE;

    是没有问题的,因为在 T2 执行时,事务 T1 已经释放掉了 number 值为 15 的记录的锁,但是如果你先执行 T2 ,再执行 T1 ,由于 T2 已经持有了 number 值为 15 的记录的锁,事务 T1 将因为获取不到这个锁而等待。
    我们再看一个使用主键进行范围查询的例子:

SELECT * FROM hero WHERE number >= 8 LOCK IN SHARE MODE;

    这个语句的执行过程其实和我们举的上一个例子类似。也是先到聚簇索引中定位到满足 number >= 8 这个条件的第一条记录,也就是 number 值为 8 的记录,然后就可以沿着由记录组成的单向链表一路向后找,每找到一条记录,就会为其加上锁,然后判断该记录符不符合范围查询的边界条件,不过这里的边界条件比较特殊:number >= 8,只要记录不小于8就算符合边界条件,所以判断和没判断是一样一样的。最后把这条记录返回给 server 层,server 层再判断 number >= 8 这个条件是否成立,如果成立的话就发送给客户端,否则的话就结束查询。不过 InnoDB 存储引擎找到索引中的最后一条记录,也就是 Supremum 伪记录之后,在存储引擎内部就可以立即判断这是一条伪记录,不必要返回给 server 层处理,也没必要给它也加上锁(也就是说在第1步中就压根儿没给这条记录加锁)。整个过程会给 number 值为8、15、20这三条记录加上 S型正经记录锁 。
(2)
    使用 👉SELECT … FOR UPDATE👈 语句来为记录加锁:
和SELECT … FOR UPDATE语句类似,只不过加的是 X型正经记录锁
(3)
    使用 👉UPDATE👈 …来为记录加锁,比方说:

UPDATE hero SET country = '汉' WHERE number >= 8;

    这条 UPDATE 语句并没有更新二级索引列,加锁方式和上边所说的 SELECT … FOR UPDATE 语句一致,也就是 X型正经记录锁
(4)如果 UPDATE 语句中更新了二级索引列,比方说:

UPDATE hero SET name = 'cao曹操' WHERE number >= 8;

    这时候会首先更新聚簇索引记录,再更新对应的二级索引记录,所以加锁的步骤就是:

  • 为number值为8的聚簇索引记录加上 X型正经记录锁
  • 然后为上一步中的记录索引记录对应的 idx_name 二级索引记录加上 X型正经记录锁
  • 为 number 值为 15 的聚簇索引记录加上X型正经记录锁
  • 然后为上一步中的记录索引记录对应的 idx_name 二级索引记录加上X型正经记录锁
  • 为 number 值为 20 的聚簇索引记录加上X型正经记录锁
  • 然后为上一步中的记录索引记录对应的idx_name二级索引记录加上X型正经记录锁。

    如果是下边这个语句:

UPDATE hero SET namey = '汉' WHERE number <= 8;

    则会对 number 值为 1、3、8 聚簇索引记录以及它们对应的二级索引记录加X型正经记录锁,加锁顺序和上边语句中的加锁顺序类似,都是先对一条聚簇索引记录加锁后,再给对应的二级索引记录加锁。之后会继续对number值为15的聚簇索引记录加锁,但是随后 InnoDB 存储引擎判断它不符合边界条件,随即会释放掉该聚簇索引记录上的锁(注意这个过程中没有对 number 值为 15 的聚簇索引记录对应的二级索引记录加锁)。

(5)
使用DELETE …来为记录加锁,比方说:

DELETE FROM hero WHERE number >= 8;

DELETE FROM hero WHERE number <= 8;

    这两个语句的加锁情况和更新带有二级索引列的 UPDATE 语句一致。

2.3 对于使用二级索引进行等值查询的情况

     在 读未提交❎ 和 读提交 ✅隔离级别下,使用 普通的二级索引和 唯一二级索 引进行加锁的过程是一样的,所以我们也就不分开讨论了。
(1)
     使用 👉SELECT … LOCK IN SHARE MODE 👈来为记录加锁,比方说:

SELECT * FROM hero WHERE name = 'c曹操' LOCK IN SHARE MODE;

     这个语句的执行过程是 先通过二级索引 idx_name 定位到满足 name = 'c曹操’条件的 二级索引记录,然后进行 回表操作。所以先要对二级索引记录加 S型正经记录锁然后再给对应的聚簇索引记录加 S型正经记录锁
     idx_name 是一个普通的二级索引,到 idx_name 索引中定位到满足 name= ‘c曹操’ 这个条件的第一条记录后,就可以沿着这条记录一路向后找。可是从我们上边的描述中可以看出来,并没有对下一条二级索引记录进行加锁,这是为什么呢?这是因为Inno DB 对等值匹配的条件有特殊处理,他们规定在 InnoDB 存储引擎层 查找到当前记录的下一条记录时,在对其加锁前 就直接判断该记录是否满足等值匹配的条件,如果不满足直接返回(也就是不加锁了),否则的话需要将其加锁后再返回给 server 层。所以这里也就不需要对下一条二级索引记录进行加锁了。

     现在要介绍一个非常有趣的事情,我们假设上边这个语句在事务T1中运行,然后事务T2中运行下边一个我们之前介绍过的语句:

UPDATE hero SET name = '曹操' WHERE number = 8;

     这两个语句都是要对 number 值为 8 的聚簇索引记录和对应的二级索引记录加锁,但是不同点是加锁的顺序不一样。这个 UPDATE 语句是先对聚簇索引记录进行加锁,后对二级索引记录进行加锁,如果在不同事务中运行上述两个语句,可能发生一种贼奇妙的事情 ——
事务T2持有了聚簇索引记录的锁,事务T1持有了二级索引记录的锁。
事务T2在等待获取二级索引记录上的锁,事务T1在等待获取聚簇索引记录上的锁。
两个事务都分别持有一个锁,而且都在等待对方已经持有的那个锁,这种情况就是所谓的👤死锁👤,两个事务都无法运行下去,必须选择一个进行回滚,对性能影响比较大。
(2)
     使用👉 SELECT … FOR UPDATE 👈语句时,比如:

SELECT * FROM hero WHERE name = 'c曹操' FOR UPDATE;

    这种情况下与 SELECT … LOCK IN SHARE MODE 语句的加锁情况类似,都是给访问到的二级索引记录和对应的聚簇索引记录加锁,只不过加的是 X型正经记录锁 罢了。
(3)
     使用 UPDATE … 来为记录加锁,比方说:
    与 更新二级索引记录的 SELECT … FOR UPDATE 的加锁情况类似,不过如果被更新的列中 还有别的二级索引列的话,对应的二级索引记录也会被加锁。
(4)
     使用 👉DELETE👈 …来为记录加锁:
     与SELECT … FOR UPDATE的加锁情况类似,不过如果表中还有别的二级索引列的话,对应的二级索引记录也会被加锁。

2.4 对于使用二级索引进行范围查询的情况

(1)
     使用 👉SELECT … LOCK IN SHARE MODE👈 来为记录加锁,比方说:

SELECT * FROM hero FORCE INDEX(idx_name)  WHERE name >= 'c曹操' LOCK IN SHARE MODE;

     因为优化器会计算 使用二级索引 进行查询的成本,在成本较大时可能选择以 全表扫描 的方式来执行查询,所以我们这里使用 FORCE INDEX(idx_name) 来强制使用二级索引 idx_name 来执行查询。
     这个语句的执行过程其实是先到二级索引中定位到满足name >= ‘c曹操’ 的第一条记录,也就是 name 值为 c曹操 的记录,然后就可以沿着这条记录的链表一路向后找,从二级索引 idx_name 的示意图中可以看出,所有的用户记录都满足name >= 'c曹操’的这个条件,所以所有的二级索引记录都会被加S型正经记录锁它们对应的聚簇索引记录也会被加S型正经记录锁不过需要注意一下加锁顺序,对一条二级索引记录加锁完后,会接着对它相应的聚簇索引记录加锁,完后才会对下一条二级索引记录进行加锁,以此类推~ 画个图表示一下就是这样:
在这里插入图片描述
再来看下边这个语句:

SELECT * FROM hero FORCE INDEX(idx_name) WHERE name <= 'c曹操' LOCK IN SHARE MODE;

     这个语句的加锁情况就有点儿有趣了。前边说在使用number <= 8 这个条件的语句中,需要把 number 值为 15 的记录也加一个锁,之后又判断它不符合边界条件而把锁释放掉。而对于查询条件 name <= ‘c曹操’ 的语句来说,执行该语句需要使用到二级索引,而与二级索引相关的条件是可以使用索引条件下推这个特性的。InnoDB规定,如果一条记录不符合索引条件下推中的条件的话,直接跳到下一条记录(这个过程根本不将其返回到server层),如果这已经是最后一条记录,那么直接向 server 层报告查询完毕。但是这里头有个问题呀:先对一条记录加了锁,然后再判断该记录是不是符合索引条件下推的条件,如果不符合直接跳到下一条记录或者直接向server层报告查询完毕,这个过程中并没有把那条被加锁的记录上的锁释放掉呀!!!。本例中使用的查询条件是 name <= ‘c曹操’,在为 name 值为’c曹操’ 的二级索引记录以及它对应的聚簇索引加锁之后,会接着二级索引中的下一条记录,也就是 name值为’l刘备’ 的那条二级索引记录,由于该记录不符合索引条件下推的条件,而且是范围查询的最后一条记录,会直接向 server 层报告查询完毕,重点是这个过程中并不会释放 name值为’l刘备’ 的二级索引记录上的锁,也就导致了语句执行完毕时的加锁情况如下所示:
在这里插入图片描述
     这样子会造成一个尴尬情况,假如 T1 执行了上述语句并且尚未提交,T2 再执行这个语句:

SELECT * FROM hero WHERE name = 'l刘备' FOR UPDATE;

     T2 中的语句需要获取 name 值为l刘备 的二级索引记录上的 X型正经记录锁,而 T1 中仍然持有 name值为l刘备 的二级索引记录上的 S型正经记录锁,这就造成了 T2获取不到锁而进入等待状态。

小贴士: 为啥不能释放不符合索引条件下推中的条件的二级索引记录上的锁呢?这个问题我也没想明白,人家就是这么规定的,我使用的MySQL版本是5.7.21,不保证其他版本中的加锁情景是否完全一致。

(2)
     使用👉SELECT … FOR UPDATE👈 语句时:
     和SELECT … FOR UPDATE语句类似,只不过加的是X型正经记录锁
(3)
     使用 👉UPDATE 👈…来为记录加锁,比方说:

UPDATE hero SET country = '汉' WHERE name >= 'c曹操';

小贴士: FORCE INDEX只对SELECT语句起作用,UPDATE语句虽然支持该语法,但实质上不起作用,DELETE语句压根儿不支持该语法。

     假设该语句执行时使用了 idx_name 二级索引来进行锁定读,那么它的加锁方式和上边所说的 SELECT … FOR UPDATE 语句一致。如果有其他二级索引列也被更新,那么也会为对应的二级索引记录进行加锁,就不赘述了。不过还有一个有趣的情况,比方说:

UPDATE hero SET country = '汉' WHERE name <= 'c曹操';

     我们前边说的索引条件下推这个特性只适用于SELECT 语句,也就是说 UPDATE 语句中无法使用,那么这个语句就会为name值为’c曹操’和’l刘备’ 的二级索引记录以及它们对应的聚簇索引进行加锁,之后在判断边界条件时发现 name 值为’l刘备’ 的二级索引记录不符合name <= ‘c曹操’ 条件,再把该二级索引记录和对应的聚簇索引记录上的锁释放掉。
(4)
使用 👉DELETE👈 …来为记录加锁,比方说:

DELETE FROM hero WHERE name >= 'c曹操';

DELETE FROM hero WHERE name <= 'c曹操';

     如果这两个语句采用二级索引来进行锁定读,那么它们的加锁情况和更新带有二级索引列的UPDATE语句一致。

2.5 全表扫描

(1) 👉LOCK IN SHARE MODE👈
     比方说:

SELECT * FROM hero WHERE country  = '魏' LOCK IN SHARE MODE;

     由于 country 列上未建索引,所以只能采用全表扫描的方式来执行这条查询语句,存储引擎每读取一条聚簇索引记录,就会为这条记录加锁一个S型正常记录锁,然后返回给 server 层,如果 server 层判断 country = ‘魏’ 这个条件是否成立,如果成立则将其发送给客户端,否则会释放掉该记录上的锁

(2)
     使用 👉SELECT … FOR UPDATE👈 进行加锁的情况与上边类似,只不过加的是 X型正经记录锁,就不赘述了。

(3)
对于 👉UPDATE👈 …和 👉DELETE👈 …的语句来说,在遍历聚簇索引中的记录,都会为该聚簇索引记录加上X型正经记录锁,然后:
     如果该聚簇索引记录不满足条件,直接把该记录上的锁释放掉。
     如果该聚簇索引记录满足条件,则会对相应的二级索引记录加上X型正经记录锁(DELETE语句会对所有二级索引列加锁,UPDATE语句只会为更新的二级索引列对应的二级索引记录加锁)。

可重复读👯‍♀️隔离级别下

    采用加锁的方式解决并发事务产生的问题时,可重复读👯‍♀️ 隔离级别与 读未提交❎、读提交✅ 这两个隔离级别相比,最主要的就是要解决幻读问题幻读问题的解决还得靠我们之前讲过的 gap锁

2.1 对于使用主键进行等值查询的情况

    使用SELECT … LOCK IN SHARE MODE来为记录加锁,比方说:

SELECT * FROM hero WHERE number = 8 LOCK IN SHARE MODE;

    我们知道主键具有唯一性,如果 在一个事务中 第一次执行上述语句时 将得到的结果集中包含一条记录,第二次执行上述语句前肯定不会有别的事务插入多条 number值为8 的记录(主键具有唯一性),也就是说一个事务中两次执行上述语句并不会发生幻读,这种情况下,我们只需要为这条number值为8的记录加一个S型正经记录锁就好了。

    但是如果我们要查询主键值不存在的记录,比方说:

SELECT * FROM hero WHERE number = 7 LOCK IN SHARE MODE;

    由于 number 值为 7 的记录不存在,为了禁止幻读现象(也就是避免在同一事务中下一次执行相同语句时得到的结果集中包含number值为7的记录),在当前事务提交前我们需要预防别的事务插入number值为7 的新记录,所以需要在 number值为8 的记录上加一个 gap锁,也就是不允许别的事务插入number值在(3, 8)这个区间的新记录。画
    如果在 读未提交❎、读提交✅ 隔离级别下 同样查询了一条 主键值不存在的记录,那么什么锁也不需要加,因为它们并不需要禁止幻读问题。
???
    其余语句使用主键进行等值查询的情况与 READ UNCOMMITTED/READ COMMITTED隔离级别下的情况类似,这里就不赘述了。

2.2 对于使用主键进行范围查询的情况

(1)
    使用 👉SELECT … LOCK IN SHARE MODE 👈语句来为记录加锁,比方说:

SELECT * FROM hero WHERE number >= 8 LOCK IN SHARE MODE;

    因为要解决 幻读问题,所以需要禁止别的事务插入number 值符合 number >= 8 的记录,又因为主键本身就是唯一的,所以我们不用担心在 number 值为 8 的前边有新记录插入,只需要保证不要让新记录插入到number值为 8 的后边就好了,所以:

  • 为number值为8的聚簇索引记录加一个S型正经记录锁
  • 为number值大于8的所有聚簇索引记录都加一个S型 next-key 锁(包括Supremum伪记录)。

小贴士: 为什么不给Supremum记录加gap锁,而要加next-key锁呢?其实设计InnoDB的大叔在处理Supremum记录上加的next-key锁时就是当作gap锁看待的,只不过为了节省锁结构(我们前边说锁的类型不一样的话不能被放到一个锁结构中)才这么做的而已,大家不必在意。

    与 读未提交❎、读提交✅ 类似,在 可重复读👯‍♀️ 隔离级别下,下边这个范围查询也是有点特殊:

SELECT * FROM hero WHERE number <= 8 LOCK IN SHARE MODE;

    这个语句的执行过程我们在之前唠叨过,在读未提交❎、读提交✅ 隔离级别下,这个语句会为 number值为1、3、8、15 这 4条记录都加上 S型正经记录锁,然后由于 number 值为 15 的记录不满足边界条件 number <= 8 ,随后便把这条记录的锁释放掉。在 可重复读👯‍♀️ 隔离级别下的加锁过程与之类似,不过会为1、3、8、15这4条记录都加上S型next-key锁,但是有一点需要十分注意:可重复读👯‍♀️隔离级别下**,在判断number值为15的记录不满足边界条件 number <= 8 后,并不会去释放加在该记录上的锁**!!!
    这样如果别的事务想要插入的新记录的number值在(-∞, 1)、(1, 3)、(3, 8)、(8, 15)之间的话,是会进入等待状态的。

小贴士: 很显然这么粗暴的做法导致的一个后果就是别的事务竟然不允许插入 number 值在 (8, 15) 这个区间中的新记录,甚至不允许别的事务再获取number 值为 15 的记录上的锁,而理论上只需要禁止别的事务插入 number值在 (-∞, 8) 之间的新记录就好。

(2)
    使用 👉SELECT … FOR UPDATE👈 语句来为记录加锁:
    和 SELECT … LOCK IN SHARE MODE 语句类似,只不过需要将上边提到的S型next-key锁替换成X型next-key锁

(3)
👉使用UPDATE 👈…来为记录加锁:
    如果UPDATE语句未更新二级索引列,比方说:

UPDATE hero SET country = '汉' WHERE number >= 8;

    这条 UPDATE 语句并没有更新二级索引列,加锁方式和上边所说的SELECT … FOR UPDATE语句一致。

    如果UPDATE语句中更新了二级索引列,比方说:

UPDATE hero SET name = 'cao曹操' WHERE number >= 8;

     对聚簇索引记录加锁的情况和 SELECT … FOR UPDATE 语句一致,也就是对 number 值为 8 的聚簇索引记录加 X型正经记录锁,对 number值15 、20的聚簇索引记录以及 Supremum 记录加 X型next-key锁。但是因为也要更新二级索引idx_name,所以也会对number值为 8、15、20 的聚簇索引记录对应的idx_name二级索引记录加X型正经记录锁。

     如果是下边这个语句:

UPDATE hero SET name = 'cao曹操' WHERE number <= 8;

     则会对number值为1、3、8、15的聚簇索引记录加X型next-key,其中 number 值为 15 的聚簇索引记录不满足 number <= 8 的边界条件,虽然在 可重复读👯‍♀️ 隔离级别下不会将它的锁释放掉,但是也并不会对这条聚簇索引记录对应的二级索引记录加锁,也就是说只会为number值为1、3、8的聚簇索引记录对应的 idx_name二级索引记录加X型正经记录锁。
(4)
使用👉DELETE 👈 …来为记录加锁,比方说:

DELETE FROM hero WHERE number >= 8;

DELETE FROM hero WHERE number <= 8;

这两个语句的加锁情况和更新带有二级索引列的UPDATE语句一致。

2.3 对于使用唯一二级索引进行等值查询的情况

     由于hero表并没有唯一二级索引,我们把原先的idx_name 修改为一个唯一二级索引uk_name:
(1)
     使用 👉SELECT … LOCK IN SHARE MODE👈 语句来为记录加锁,比方说:

SELECT * FROM hero WHERE name = 'c曹操' LOCK IN SHARE MODE;

     由于 唯一二级索引 具有唯一性,如果在一个事务中第一次执行上述语句时将得到一条记录,第二次执行上述语句前肯定不会有别的事务插入多条 name 值为 ‘c曹操’ 的记录(二级索引具有唯一性),也就是说一个事务中两次执行上述语句并不会发生幻读,这种情况下和READ UNCOMMITTED/READ COMMITTED隔离级别下一样,我们只需要为这条name值为’c曹操’的二级索引记录加一个S型正经记录锁然后再为它对应的聚簇索引记录加一个S型正经记录锁就好了
     注意加锁顺序,是先对二级索引记录加锁,再对聚簇索引加锁。

     如果对唯一二级索引列进行等值查询的记录并不存在,比如:

SELECT * FROM hero WHERE name = 'g关羽' LOCK IN SHARE MODE;

     为了禁止幻读,所以需要保证别的事务不能再插入name值为’g关羽’的新记录。在唯一二级索引 uk_name中,键值比’g关羽’大的第一条记录的键值为 l刘备,所以需要在这条二级索引记录上加一个gap锁。
     注意,这里只对二级索引记录进行加锁,并不会对聚簇索引记录进行加锁。
(2)
使用 👉SELECT … FOR UPDATE👈 语句来为记录加锁:
和SELECT … LOCK IN SHARE MODE语句类似,只不过加的是X型正经记录锁。
(3)
使用👉UPDATE 👈…来为记录加锁:
     与 SELECT … FOR UPDATE 的加锁情况类似,不过如果被更新的列中还有别的二级索引列的话,这些对应的二级索引记录也会被加X型正经记录锁
(4)
使用👉DELETE 👈…来为记录加锁:
     与SELECT … FOR UPDATE的加锁情况类似,不过如果表中还有别的二级索引列的话,这些对应的二级索引记录也会被加X型正经记录锁。

2.4 对于使用唯一二级索引进行范围查询的情况

(1)
使用👉SELECT … LOCK IN SHARE MODE👈语句来为记录加锁,比方说:

SELECT * FROM hero FORCE INDEX(uk_name) WHERE name >= 'c曹操' LOCK IN SHARE MODE;

     这个语句的执行过程其实是先到 二级索引中定位到满足name >= ‘c曹操’ 的第一条记录,也就是 name值为c曹操 的记录,然后就可以沿着由记录组成的单向链表一路向后找。从二级索引 idx_name 的示意图中可以看出,所有的用户记录都满足name >= 'c曹操’的这个条件,所以所有的二级索引记录都会被加S型next-key锁它们对应的聚簇索引记录也会被加S型正经记录锁二级索引的最后一条Supremum记录也会被加S型next-key锁。不过需要注意一下加锁顺序,对一条二级索引记录加锁完后,会接着对它响应的聚簇索引记录加锁,完后才会对下一条二级索引记录进行加锁,以此类推~
     稍等一下,不是说uk_name是唯一二级索引么?唯一二级索引本身就能保证其自身的值是唯一的,那为啥还要给name值为’c曹操’的记录加上S型next-key锁,而不是S型正经记录锁呢?其实我也不知道,按理说只需要给这条二级索引记录加S型正经记录锁就好了。
再来看下边这个语句:

SELECT * FROM hero WHERE name <= 'c曹操' LOCK IN SHARE MODE;

     这个语句先会为 name值为 ‘c曹操’ 的二级索引记录加S型next-key 锁以及它对应的聚簇索引记录加S型正经记录锁。然后还要给name值为’l刘备’的二级索引记录加S型next-key锁,name值为’l刘备’的二级索引记录不满足索引条件下推的name <= 'c曹操’条件,压根儿不会释放掉该记录的锁就直接报告server层查询完毕了。这样可以禁止其他事务插入name值在(‘c曹操’, ‘l刘备’)之间的新记录,从而防止幻读产生。
     这里大家要注意一下,设计InnoDB的大叔在这里给name值为’l刘备’的二级索引记录加的是S型next-key锁,而不是简单的gap锁。
(2)
使用 👉SELECT … FOR UPDATE👈 语句来为记录加锁:
     和SELECT … LOCK IN SHARE MODE语句类似,只不过加的是X型正经记录锁。
(3)
使用👉UPDATE👈 …来为记录加锁,比方说:

UPDATE hero SET country = '汉' WHERE name >= 'c曹操';

     假设该语句执行时使用了 uk_name 二级索引来进行锁定读(如果二级索引扫描的记录太多,也可能因为成本过大直接使用全表扫描的方式进行锁定读),而这条 UPDATE 语句并没有更新二级索引列,那么它的加锁方式和上边所说的 SELECT … FOR UPDATE 语句一致。如果有其他二级索引列也被更新,那么也会为这些二级索引记录进行加锁,就不赘述了。不过还需要强调一种情况,比方说:

UPDATE hero SET country = '汉' WHERE name <= 'c曹操';

     我们前边说的索引条件下推这个特性只适用于SELECT 语句,也就是说 UPDATE 语句中无法使用,无法使用索引条件下推这个特性时需要先进行回表操作,那么这个语句就会为name值为’c曹操’和’l刘备’的二级索引记录加X型next-key锁,对它们对应的聚簇索引记录进行加X型正经记录锁。不过之后在判断边界条件时,虽然name值为’l刘备’的二级索引记录不符合name <= 'c曹操’的边界条件,但是在可重复读👯‍♀️隔离级别下并不会释放该记录上加的锁。

(4)
使用DELETE …来为记录加锁,比方说:

DELETE FROM hero WHERE name >= 'c曹操';

DELETE FROM hero WHERE name <= 'c曹操';

     如果这两个语句采用二级索引来进行锁定读,那么它们的加锁情况和更新带有二级索引列的 UPDATE 语句一致,就不画图了。

2.5 对于使用普通二级索引进行等值查询的情况

(1)
使用👉SELECT … LOCK IN SHARE MODE👈语句来为记录加锁,比方说:

SELECT * FROM hero WHERE name = 'c曹操' LOCK IN SHARE MODE;

     由于普通的二级索引没有唯一性,所以一个事务在执行上述语句之后,要阻止别的事务插入name值为’c曹操’的新记录,设计InnoDB的大叔采用下边的方式对上述语句进行加锁:

  • 对所有name值为’c曹操’的二级索引记录加S型next-key锁,它们对应的聚簇索引记录加S型正经就锁。
  • 对最后一个name值为’c曹操’的二级索引记录的下一条二级索引记录加gap锁。

     如果对普通二级索引等值查询的值并不存在,比如:

SELECT * FROM hero WHERE name = 'g关羽' LOCK IN SHARE MODE;

     加锁方式和我们上边唠叨过的唯一二级索引的情况是一样的,就不赘述了。
(2)
使用👉SELECT … FOR UPDATE👈语句来为记录加锁,比如:
和SELECT … LOCK IN SHARE MODE语句类似,只不过加的是X型正经记录锁。
(3)
使用👉UPDATE …👈来为记录加锁:
     与SELECT … FOR UPDATE的加锁情况类似,不过如果被更新的列中还有别的二级索引列的话,这些对应的二级索引记录也会被加锁。
(4)
使用👉DELETE👈 …来为记录加锁:
     与 SELECT … FOR UPDATE 的加锁情况类似,不过如果表中还有别的二级索引列的话,这些对应的二级索引记录也会被加锁。

2.6 对于使用普通二级索引进行范围查询的情况

与唯一二级索引的加锁情况类似,就不多唠叨了哈~

2.7 全表扫描的情况

(1)
比方说:

SELECT * FROM hero WHERE country  = '魏' LOCK IN SHARE MODE;

     由于country列上未建索引,所以只能采用全表扫描的方式来执行这条查询语句,存储引擎每读取一条聚簇索引记录,就会为这条记录加锁一个S型next-key锁,然后返回给 server 层,如果 server 层判断 country = ‘魏’ 这个条件是否成立,如果成立则将其发送给客户端,否则会向 InnoDB存储引擎 发送释放掉该记录上的锁的消息,不过在可重复读👯‍♀️ 下,InnoDB存储引擎并不会真正的释放掉锁,所以聚簇索引的全部记录都会被加锁,并且在事务提交前不释放。

     全部记录都被加了 next-key锁!此时别的事务别说想向表中插入啥新记录了,就是对某条记录加X锁 都不可以,这种情况下会极大影响访问该表的并发事务处理能力,所以如果可能的话,尽可能为表建立合适的索引吧~
(2)
     使用👉 SELECT … FOR UPDATE👈 进行加锁的情况与上边类似,只不过加的是X型正经记录锁 ,就不赘述了。
(3)
     对于 UPDATE … 语句来说,加锁情况与 SELECT … FOR UPDATE 类似,不过如果被更新的列中还有别的二级索引列的话,这些对应的二级索引记录也会被加X型正经记录锁。
(4)
    DELETE …的语句来说,加锁情况与SELECT … FOR UPDATE类似,不过如果表中还有别的二级索引列的话,这些对应的二级索引记录也会被加X型正经记录锁。

3.INSERT语句

     INSERT语句一般情况下不加锁,不过当前事务在插入一条记录前需要先定位到该记录在B+树中的位置,如果该位置的下一条记录已经被加了gap锁(next-key锁也包含gap锁,之后就不强调了),那么当前事务会在该记录上加上一种类型为插入意向锁的锁,并且事务进入等待状态。
下边要看的是两种INSERT语句遇到的特殊情况:

3.1 遇到重复键(duplicate key)

     在插入一条新记录时,首先要做的事情其实是定位到这条新记录应该插入到B+树的哪个位置。如果定位位置时发现了有已存在记录的主键或者唯一二级索引列与待插入记录的主键或者唯一二级索引列相同(不过可以有多条记录的唯一二级索引列的值同时为NULL),那么此时此时是会报错的。比方说我们插入新记录,该记录的主键值已经被包含在hero表中了:

mysql> BEGIN;
Query OK, 0 rows affected (0.01 sec)

mysql> INSERT INTO hero VALUES(20, 'g关羽', '蜀');
ERROR 1062 (23000): Duplicate entry '20' for key 'PRIMARY'

     当然,在生成报错信息前,其实还需要做一件非常重要的事情 —— 对聚簇索引中number值为20的那条记录加S锁。不过具体的行锁类型在不同隔离级别下是不一样的:
(1)
在读未提交❎、读提交✅ 隔离级别下,加的是S型正经记录锁
(2)
在可重复读👯‍♀️ / 串行读 ♒E隔离级别下,加的是S型next-key锁
(3)
如果是唯一二级索引列值重复,

mysql> INSERT INTO hero VALUES(30, 'c曹操', '魏');
ERROR 1062 (23000): Duplicate entry 'c曹操' for key 'uk_name'

    很显然,hero表中之前就包含name值为’c曹操’的记录,如果再插入一条 name值为’c曹操’ 的新记录时,虽然插入对应的聚簇索引记录没问题,但是在插入uk_name唯一二级索引记录时便会报错,不过在报错之前还是会把name值为’c曹操’那条二级索引记录加一个S锁。需要注意的是,不管是哪个隔离级别,针对在插入新记录时遇到重复的唯一二级索引列的情况,会对已经在B+树中的唯一二级索引记录加next-key锁。

    另外,如果我们使用的是INSERT … ON DUPLICATE KEY …这样的语法来插入记录时,如果遇到主键或者唯一二级索引列值重复的情况,会对 B+ 树中已存在的相同键值的记录加X锁,而不是S锁。

3.2 外键检查

    大家别忘了MySQL还是一个支持外键的数据库,比方说我们再为三国英雄的战马建一个表:

CREATE TABLE horse (
    number INT PRIMARY KEY,
    horse_name VARCHAR(100),
    FOREIGN KEY (number) REFERENCES hero(number)
)Engine=InnoDB CHARSET=utf8;

    这样 hero 表就算是一个父表,新建的 horse 表就算一个子表,其中horse表的number列是参照hero表的number列。现在如果我们向子表中插入一条记录时:
(1)
如果待插入记录的number值在hero表中能找到。

    比方说我们为horse表中新插入的记录的number值为8,而在hero表中number值为8的记录代表曹操,他的马是绝影:

mysql> BEGIN;
Query OK, 0 rows affected (0.00 sec)

mysql> INSERT INTO horse VALUES(8, '绝影');
Query OK, 1 row affected (5 min 58.04 sec)

    在插入成功之前,不论当前事务的隔离级别是什么,只需要直接给父表hero的number值为3的记录加一个S型正经记录锁

(2)
如果待插入记录的 number 值在 hero 表中找不到。
    比方说我们为 horse 表中新插入的记录的 number 值为 2 ,而在 hero 表中不存在 number 值为 2 的记录:

mysql> BEGIN;
Query OK, 0 rows affected (0.00 sec)

mysql> INSERT INTO horse VALUES(5, '绝影');
Query OK, 1 row affected (5 min 58.04 sec)

虽然插入失败了,但是这个过程中需要对父表hero的某些记录进行加锁:

  • 在读未提交❎、读提交✅隔离级别下,并不对记录加锁。
  • 在可重复读👯‍♀️/串行读 ♒隔离级别下,加的是gap锁。
  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值