美文网首页
MySQL系列之二 -- InnoDB所使用的七种锁

MySQL系列之二 -- InnoDB所使用的七种锁

作者: 花神子 | 来源:发表于2020-02-07 17:29 被阅读0次

上一篇文章[MySQL系列之一 -- 事务](https://www.jianshu.com/p/5b213becd8ad)中在介绍事务的隔离等级实现原理是引入了MySQL锁的机制,本篇文章就针对MySQL锁的机制来做下简单介绍,总的来说,InnoDB共有七种类型的锁

  1. 自增锁(Auto-inc Locks)
  2. 共享/排它锁(Shared and Exclusive Locks)
  3. 意向锁(Intention Locks)
  4. 插入意向锁(Insert Intention Locks)
  5. 记录锁(Record Locks)
  6. 间隙锁(Gap Locks)
  7. 临键锁(Next-key Locks)

1. 自增锁(Auto-inc Locks)

说明

自增锁是一种特殊的表级别锁(table-level lock),专门针对事务插入AUTO_INCREMENT类型的列。
最简单的情况,如果一个事务正在往表中插入记录,所有其他事务的插入必须等待,以便第一个事务插入的行,是连续的主键值。

_An AUTO-INC lock is a special table-level lock taken by transactions inserting into tables with 
AUTO_INCREMENT columns. In the simplest case, if one transaction is inserting values into 
the table, any other transactions must wait to do their own inserts into that table, so that rows 
inserted by the first transaction receive consecutive primary key values._

与此同时,InnoDB提供了innodb_autoinc_lock_mode配置,可以调节与改变该锁的模式与行为。

2. 共享/排它锁(Shared and Exclusive Locks)

  • 事务拿到某一行记录的共享S锁,才可以读取这一行;多个事务可以拿到一把S锁,读读可以并行;
  • 事务拿到某一行记录的排它X锁,才可以修改或者删除这一行;只有一个事务可以拿到X锁,写写/读写必须互斥;
  • 兼容互斥表如下:
      S      X
S    兼容    互斥
X    互斥    互斥

3 意向锁(Intention Locks)

意向锁是指,未来的某个时刻,事务可能要加共享/排它锁了,先提前声明一个意向。
InnoDB支持多粒度锁(multiple granularity locking),它允许行级锁与表级锁共存,实际应用中,InnoDB使用的是意向锁。

意向锁有这样一些特点:
(1)首先,意向锁,是一个表级别的锁(table-level locking);
(2)意向锁分为:

  • 意向共享锁(intention shared lock, IS),它预示着,事务有意向对表中的某些行加共享S锁
  • 意向排它锁(intention exclusive lock, IX),它预示着,事务有意向对表中的某些行加排它X锁

举个例子:
select ... lock in share mode,要设置IS锁
select ... for update,要设置IX锁

(3)意向锁协议(intention locking protocol)并不复杂:

  • 事务要获得某些行的S锁,必须先获得表的IS锁
  • 事务要获得某些行的X锁,必须先获得表的IX锁

(4)由于意向锁仅仅表明意向,它其实是比较弱的锁,意向锁之间并不相互互斥,而是可以并行,其兼容互斥表如下:

      IS       IX
IS   兼容      兼容
IX   兼容      兼容

(5)额,既然意向锁之间都相互兼容,那其意义在哪里呢?它会与共享锁/排它锁互斥,其兼容互斥表如下:

      S        X
IS   兼容      互斥
IX   互斥      互斥

画外音:排它锁是很强的锁,不与其他类型的锁兼容。这也很好理解,修改和删除某一行的时候,必须获得强锁,禁止这一行上的其他并发,以保障数据的一致性。

4插入意向锁(Insert Intention Locks)

插入意向锁,是间隙锁(Gap Locks)的一种(所以,也是实施在索引上的),它是专门针对insert操作的。InnoDB使用插入意向锁,可以提高插入并发;
对已有数据行的修改与删除,必须加强互斥锁X锁,那对于数据的插入,是否还需要加这么强的锁,来实施互斥呢?插入意向锁,孕育而生。

它的核心是:
多个事务,在同一个索引,同一个范围区间插入记录时,如果插入的位置不冲突,不会阻塞彼此。

_Insert Intention Lock signals the intent to insert in such a way that multiple transactions 
inserting into the same index gap need not wait for each other if they are not inserting at the 
same position within the gap._
  • 虽然事务隔离级别是RR,虽然是同一个索引,虽然是同一个区间,但插入的记录并不冲突,则使用的是插入意向锁, 并不会阻塞事务B

5 记录锁(Record Locks)

记录锁,它封锁索引记录。例如:

select * from t where id=1 for update;

它会在id=1的索引记录上加锁,以阻止其他事务插入,更新,删除id=1的这一行。
注意

select * from t where id=1;

快照读(SnapShot Read),它并不加锁.

另: RC级别下,采用记录锁,解决脏读问题,但是会出现幻读,以及不可重复读。

6 间隙锁(Gap Locks)

间隙锁,它封锁索引记录中的间隔,或者第一条索引记录之前的范围,又或者最后一条索引记录之后的范围。
间隙锁的主要目的,就是为了防止其他事务在间隔中插入数据,以导致“不可重复读”。
例如:

select * from t 
    where id between 8 and 15 
    for update;

注意: 如果把事务的隔离级别降级为读提交(Read Committed, RC),间隙锁则会自动失效,所以间隙锁对应的隔离级别是RR。
另: RR级别下其实Next-Key Lock的方式对数据行进行加锁,并不是仅仅使用间隙锁。

临键锁(Next-Key Locks)

临键锁,是记录锁与间隙锁的组合,它的封锁范围,既包含索引记录,又包含索引区间。
临键锁的主要目的,也是为了避免幻读(Phantom Read)。如果把事务的隔离级别降级为RC,临键锁则也会失效。

默认情况下,InnoDB工作在可重复读(Repeatable Read)隔离级别下,并且会以Next-Key Lock的方式对数据行进行加锁,这样可以有效防止幻读的发生。Next-Key Lock是行锁和间隙锁的组合,当InnoDB扫描索引记录的时候,会首先对索引记录加上行锁(Record Lock),再对索引记录两边的间隙加上间隙锁(Gap Lock)。加上间隙锁之后,其他事务就不能在这个间隙修改或者插入记录。

相关文章

  • MySQL系列之二 -- InnoDB所使用的七种锁

    上一篇文章[MySQL系列之一 -- 事务](https://www.jianshu.com/p/5b213bec...

  • 资深架构大佬分享:MySQL InnoDB 存储引擎大观

    MySQL InnoDB 引擎现在广为使用,它提供了事务,行锁,日志等一系列特性,本文分析下 InnoDB的内部实...

  • MySQL InnoDB存储引擎大观

    MySQL InnoDB 引擎现在广为使用,它提供了事务,行锁,日志等一系列特性,本文分析下 InnoDB的内部实...

  • Mysql

    MySQL InnoDB中使用悲观锁 要使用悲观锁,我们必须关闭mysql数据库的自动提交属性,因为MySQL默认...

  • MySQL数据库InnoDB引擎锁定范围以及sql语句使用的锁类

    1.InnoDB引擎使用了七种类型的锁,他们分别是: 共享锁(S锁)/排它锁(X锁) 意向锁(Intention ...

  • 淘宝MySQL文档整理

    MySQL · 引擎特性 · InnoDB 事务锁系统简介 MySQL · 引擎特性 · Innodb 锁子系统浅...

  • Innodb的锁

    Innodb的锁是行级锁 mysql delete是否会锁表 MySQL的InnoDB存储引擎支持行级锁,Inno...

  • InnoDB的锁

    本章节主要讨论InnoDB所使用到的锁的类型。注:本文主要参考Mysql 5.7的官方文档,如果没有特别声明,本文...

  • 在kotlin中使用mysql行级锁

    mysql中的锁 首先需要介绍一下mysql的锁。一般我们使用InnoDB数据库引擎+行级锁,SQL为:SELEC...

  • mysql规范

    一、基础规范【强制】使用InnoDB存储引擎解读:InnoDB存储引擎是MySQL默认存储引擎,支持事务和行级锁,...

网友评论

      本文标题:MySQL系列之二 -- InnoDB所使用的七种锁

      本文链接:https://www.haomeiwen.com/subject/smqvxhtx.html