MySQL中的各种锁(行锁、间隙锁、临键锁等等LBCC)

article/2025/9/15 3:54:27

目录

  • 1.快照读和锁定读
    • 1.1 一致性读 / 快照读
    • 1.2 锁定读
      • 1.2.1 共享锁和独占锁
      • 1.2.2 锁定读的语句
        • 1.2.2.1 Lock In Share Mode 对记录加S共享锁
        • 1.2.2.2 For Update 对记录加X独占锁
  • 2.表锁和行锁、X锁/S锁、IX锁、IS锁
    • 2.1 意向锁 Intention Lock
    • 2.2 InnoDB的表锁
  • 3.InnoDB的行锁
    • 3.1 Record Lock 记录锁
    • 3.2 Gap Lock 间隙锁
    • 3.3 Next-Key Lock 临键锁
    • 3.4 Insert Intention Lock 插入意向锁(无实用)
  • 4.锁的内存结构

1.快照读和锁定读

1.1 一致性读 / 快照读

事务的读取操作如果使用MVCC的方式读取就是快照读,它是一种无锁的读。所有的SELECT查询语句在可重复读、读已提交的级别下都是快照读。由于一致性读不会对记录进行加锁的操作,所以在并发环境下其它事务可以对表中的记录进行写操作。

1.2 锁定读

1.2.1 共享锁和独占锁

        - `共享锁`:简称`S锁`,事务要读取一条记录时需要获取该记录的S锁。读读允许。- `独占锁/排它锁`:简称`X锁`,事务要改动一条记录需要获得该记录的X锁。读写、写读、写写都会阻塞。类似于Java中的ReentrantReadWriteLock。

1.2.2 锁定读的语句

锁定读:读取记录信息前对记录进行加锁(S/X锁都行)的行为就叫锁定读

1.2.2.1 Lock In Share Mode 对记录加S共享锁

SELECT * FROM user WHERE id = 1 LOCK IN SHARE MODE,这段语句就会对该记录进行加S锁,允许其它事务的读取,其它事务也可以继续加S锁,但是不允许其它事务加X锁,否则会阻塞,直到当前事务提交后释放了S锁,其它线程才能加X锁。

1.2.2.2 For Update 对记录加X独占锁

SELECT * FROM user WHERE id = 1 FOR UPDATE,对该记录进行加X独占锁,其它事务要加S锁或者X锁都会被阻塞。生产环境严谨使用For Update。

2.表锁和行锁、X锁/S锁、IX锁、IS锁

MyISAM、MEMORY这些存储引擎他们只支持表锁且不支持事务。InnoDB支持表锁和行锁。

2.1 意向锁 Intention Lock

     - `意向共享锁 IS锁`:当事务准备在某条记录上加S锁时,需要先在表上加IS锁。- `意向独占锁 IX锁`:当事务准备在某条记录上加X锁时,需要现在表上加IX锁。

2.2 InnoDB的表锁

一般情况InnoDB引擎在做CRUD时,不会对表进行加表级别的S锁或X锁的。但是对某个表执行了DDL语句就会阻塞其他的事务对表记录的修改。同样,一个正在对表记录进行修改的事务会将DDL操作阻塞。这个过程是通过在Service层的元数据锁 Metadata Lock来完成的。
Auto-Inc锁:主要用来对表进行插入记录,系统为记录赋予自增ID的作用,Insert语句执行时需要对表进行加Auto-Inc锁来保证这个递增值的分配安全,和其它锁不一样的是,这个锁在Insert结束后就会立即释放。
InnoDB对自增ID的线程安全只采用Auto-Inc锁的方式,有一个参数innodb_autoinc_lock_mode来控制是否采用Auto-Inc锁方式,如果不采用这种方式就会采用一种轻量级锁的方式来保证自增ID的准确:简单来说就是为Insert语句修饰了自增属性的列获取一把锁然后分配ID,分配完成后立即释放,无需等到Insert语句结束才释放。
一般,对于确定Insert多少条记录的场景使用轻量级锁,对于不确定数量的Insert来说采用Auto-Inc锁的方式。

3.InnoDB的行锁

InnoDB的锁都是基于索引加锁的,如果修改语句没有使用索引,那么将升级为表锁。

3.1 Record Lock 记录锁

在这里插入图片描述

最普通的记录锁,普通记录锁是区分S锁和X锁的。
使用唯一性的索引进行等值查询且精准匹配到一条记录时,例如select * from student where id =2 for update;就会将李四的记录锁定。

3.2 Gap Lock 间隙锁

间隙锁和临键锁的出现就是为了解决可重复读隔离级别下的幻读问题(快照读遇到当前读会有幻读问题)。间隙锁就是在目标记录行的上一条集合和当前目标记录之间的缝隙加一个锁。
在这里插入图片描述

仅在可重复读级别下生效。
使用等值查询/范围查询时,并且没有命中存在的记录,那么在索引对应区间生成间隙锁。例如select * from student where id =5 for update;select * from t where id > 2 and id < 7 for update;就会将(2, 7)区间锁住。
因为InnoDB的锁时基于索引的,索引是根据索引字段进行排序的,所以通过间隙锁可以来阻止其它事务向该间隙插入记录。

3.3 Next-Key Lock 临键锁

临键锁是记录锁+间隙锁的组合形式,用来保护当前记录不被修改和这个间隙不予许插入。
在这里插入图片描述

临键锁适用于非唯一性索引,且仅在可重复读级别下生效。和间隙锁不同的是,临键锁是SQL命中了部分记录的场景,除了会锁定间隙外还会锁定命中记录的下一个区间,是一个左开右闭的模式。
在可重复读级别下,MySQL默认就是使用的临键锁,当在非唯一索引下如果SQL没有命中记录则就是间隙锁,命中了记录就是临键锁。如果用唯一索引且SQL命中了记录,那么加的就是普通记录锁,否则就是间隙锁。

3.4 Insert Intention Lock 插入意向锁(无实用)

当A事务要插入一条记录时,要判断索引在这个位置区间是否被加了间隙锁或者临键锁,如果有的话,A事务就会阻塞,直到其它事务提交释放了锁为止。但是A事务处在等待过程中也需要在内存中生成一个锁结构,来表名A事务处于锁的等待过程中,这个锁就是插入意向锁。

4.锁的内存结构

在这里插入图片描述

行锁的n_bits:一个页面中存在多条记录,用不同的比特位来区分到底是哪一条记录加了锁。


http://chatgpt.dhexx.cn/article/tAdWuNQV.shtml

相关文章

什么是间隙锁?怎样避免间隙锁的危害?

间隙锁&#xff1a; 当我们用范围条件&#xff0c;而不是使用相等条件检索数据&#xff0c;并请求共享或排他锁时&#xff0c; InnoDB 会给符合条件的已有数据进行加锁&#xff1b;对于键值在条件范围内但并不存在的记录&#xff0c;叫做"间隙&#xff08; GAP )", …

MySQL 行级锁(行锁、临键锁、间隙锁) 小白教程

一、行级锁 行级锁&#xff0c;每次操作锁住对应的行数据。锁定粒度最小&#xff0c;发生锁冲突的概率最低&#xff0c;并发度最高。应用在InnoDB存储引擎中。 InnoDB的数据是基于索引组织的&#xff0c;行锁是通过对索引上的索引项加锁来实现的&#xff0c;而不是对记录加的锁…

MySQL 啥时候用记录锁,啥时候用间隙锁?

MySQL 啥时候会用记录锁&#xff0c;啥时候会用间隙锁&#xff0c;啥时候又会用 Next-Key 锁呢&#xff1f;今天我们就来做一些测试&#xff0c;弄清楚这个问题。 文章思维导图 影响因素 在开始之前&#xff0c;我们需要声明的是&#xff1a;本文所有测试及结论的前提均是在「…

MySQL的间隙锁

一、、为什么要引入间隙锁 MySQL引入间隙锁&#xff08;Gap Lock&#xff09;&#xff0c;是为了在可重复读事务隔离级别中&#xff0c;解决幻读问题锁引入的锁机制。 二、幻读 2.1、什么是幻读 幻读是指&#xff0c;当一个事务先后两次查询同一个范围的时候&#xff0c;查…

MySQL间隙锁(幻读解决原理)

文章目录 一、间隙锁概念二、测试间隙锁范围加锁场景1&#xff1a;用不可重复的主键id测试间隙锁场景2&#xff1a;用可重复的age&#xff08;有索引&#xff09;测试间隙锁场景3&#xff1a;实际情况需要具体分析用的到底是行锁还是表锁 三、测试等值间隙锁1. 测试不能重复的主…

mysql隔离级别RR下的行锁、临键锁、间隙锁详解及运用

一&#xff1a;mysql 锁的基本概念 锁&#xff1a;悲观锁、乐观锁 悲观锁&#xff1a;写锁 for update、读锁for share 写锁&#xff1a;只允许当前事务读写&#xff0c;其它事务全部等待&#xff0c;包括读取数据&#xff0c;锁的数据范围需要具体分析 读锁&#xff1a;允…

MySQL行锁、表锁、间隙锁详解

前言 我们前几篇讲了索引是什么&#xff0c;如何使用explain分析索引使用情况&#xff0c;如何去优化索引&#xff0c;以及show profiles分析SQL语句执行资源消耗的学习。今天我们来讲讲MySQL的各种锁&#xff0c;这里存储引擎我们使用InnoDB&#xff1b; 准备工作 创建表 t…

Mysql_行锁、临键锁、间隙锁的理解

目录 行锁间隙锁临键锁总结 行锁 行锁&#xff0c;也称为记录锁。 当我们针对主键或者唯一索引加锁的时候&#xff0c;Mysql默认会对查询的这一行数据加行锁&#xff0c;避免其他事务对这一行数据进行修改。 间隙锁 间隙锁&#xff0c;顾名思义&#xff0c;就是锁定一个索引…

【MySQL】MySQL的锁机制 - 记录锁、间隙锁、临键锁

目录 一、参考链接二、总结1、间隙锁的目的2、什么时候产生间隙锁&#xff1f;3、间隙锁锁定的范围4、需要注意的点&#xff08;重要&#xff09; 三、案例建表案例1:查询已存在的单条记录案例2:查询不存在的记录案例3:查询多条记录&#xff08;范围查询&#xff09; 四、其余验…

MySQL的锁机制 - 记录锁、间隙锁、临键锁

记录锁(Record Locks) 记录锁是 封锁记录&#xff0c;记录锁也叫行锁&#xff0c;例如&#xff1a; SELECT * FROM test WHERE id1 FOR UPDATE; 它会在 id1 的记录上加上记录锁&#xff0c;以阻止其他事务插入&#xff0c;更新&#xff0c;删除 id1 这一行。 记录锁、间隙锁…

行锁、间隙锁、next-key锁

参考博客&#xff1a;cmysql锁&#xff08;九&#xff09;innodb下的记录锁&#xff0c;间隙锁&#xff0c;next-key锁 - 简书 Mysql 行锁、间隙锁和next-key锁详解_程序员掉头发的博客-CSDN博客_行锁 间隙锁 行锁(Record Lock)&#xff1a; 对索引记录加锁。间隙锁(Gap Lock…

mysql记录锁、间隙锁、临键锁

名词解释 记录锁&#xff1a;record lock,即锁住一条记录 间隙锁&#xff1a;gap lock,即锁定一个区间&#xff0c;左开又开 临键锁&#xff1a;记录锁间隙锁锁定的区间&#xff0c;左开右闭 mysql如何加锁 假设有如下数据 其中&#xff0c;id 是主键索引&#xff08;唯一索…

【MySQL】记录锁?间隙锁?临键锁?到底锁了些什么?这一篇帮你捋清楚( ̄∇ ̄)/

特别强调&#xff5e; 本测试使用的是MySQL 8.0.27&#xff5e; 8.0.27&#xff5e; 8.0.27&#xff08;因为不同版本命令可能会有差异哈&#xff09; 打开两个终端&#xff0c;分别连接上MySQL&#xff0c;使用select global.transaction_isolation;查看隔离级别&#xff08…

MySQL的锁机制——记录锁、间隙锁、临键锁

记录锁(Record Locks) 记录锁锁住的是索引记录&#xff0c;记录锁也叫行锁。如果使用索引作为条件命中了记录&#xff0c;那么就是记录锁&#xff0c;被锁住的记录不能被别的事务插入相同的索引键值&#xff0c;修改和删除。 例如&#xff1a; select * from test_table whe…

mysql 间隙锁

一、什么是间隙锁&#xff1f; 间隙锁&#xff08;Gap Lock&#xff09;&#xff1a;锁加在不存在的空闲空间&#xff0c;可以是两个索引记录之间&#xff0c;也可能是第一个索引记录之前或最后一个索引之后的空间。 当我们用范围条件而不是相等条件索引数据&#xff0c;并请求…

什么是间隙锁

中心思想 间隙锁锁的是索引叶子节点的next指针。 意义 解决了mysql RR级别下是幻读的问题。 快照读 在RR隔离级别下&#xff1a;快照读有可能读到数据的历史版本&#xff0c;也有可能读到数据的当前版本。所以快照读无需用锁也不会发生幻读的情况。 当前读 当前读&#…

MySql进阶-间隙锁(gap-key)

文章目录 Innodb锁算法关闭Gap LockGap-key 解决的问题间隙锁影响MVCC 核心原理ReadView 可参考 快照读&#xff0c;当前读可参考 参考《InnoDB存储引擎》 注意&#xff1a;gap-key是innodb存储引擎来解决当前读的幻读问题的。对于隔离级别下的可重复读只能解决快照读的幻读问题…

dubbo接口调试工具

最近的项目使用来dubbo进行开发&#xff0c;虽然可以使用telne客户端t进行dubbo的接口的调试&#xff0c;但总感觉调试起来不太方便&#xff0c;并且限制太多&#xff0c;于是抽了点时间出来编写了一个dubbo的客户端可视化调试工具&#xff0c;功能虽简单但可以快速的调试dubbo…

Android Studio调试工具总结

前言&#xff1a;写代码不可避免有Bug&#xff0c;通常情况下除了日志最直接的调试手段就是debug&#xff1b;当我们的程序出现bug时&#xff0c;调试可以快速的找到bug。进入调试状态&#xff0c;我们可以清楚的了解程序的整个执行过程&#xff0c;可以对内存的数据进行监视。…

串口调试、udp 调试、tcp 调试,websocket 调试,通讯调试工具

简介 一个通讯测试工具&#xff0c;QSAK&#xff08;Qt Swiss Army Knife&#xff09;是一款基于Qt开源框架打造的多功能、跨平台调试工具。目前支持串口调试、udp 调试、tcp 调试及 websocket 调试等。支持 Windows、Linux、raspberry pi 等平台。 免费、开源、绿色、免安装…