我的编程空间,编程开发者的网络收藏夹
学习永远不晚

MySQL中有哪些加锁场景

短信预约 -IT技能 免费直播动态提醒
省份

北京

  • 北京
  • 上海
  • 天津
  • 重庆
  • 河北
  • 山东
  • 辽宁
  • 黑龙江
  • 吉林
  • 甘肃
  • 青海
  • 河南
  • 江苏
  • 湖北
  • 湖南
  • 江西
  • 浙江
  • 广东
  • 云南
  • 福建
  • 海南
  • 山西
  • 四川
  • 陕西
  • 贵州
  • 安徽
  • 广西
  • 内蒙
  • 西藏
  • 新疆
  • 宁夏
  • 兵团
手机号立即预约

请填写图片验证码后获取短信验证码

看不清楚,换张图片

免费获取短信验证码

MySQL中有哪些加锁场景

MySQL中有哪些加锁场景,很多新手对此不是很清楚,为了帮助大家解决这个难题,下面小编将为大家详细讲解,有这方面需求的人可以来学习下,希望你能有所收获。

如下图所示,数据库的隔离等级,SQL 语句和当前数据库数据会共同影响该条 SQL 执行时数据库生成的锁模式,锁类型和锁数量

MySQL中有哪些加锁场景

下面,我们会首先讲解一下隔离等级、不同 SQL 语句 和 当前数据库数据对生成锁影响的基本规则,然后再依次具体 SQL 的加锁场景。

隔离等级对加锁的影响

MySQL 的隔离等级对加锁有影响,所以在分析具体加锁场景时,首先要确定当前的隔离等级

  • 读未提交(Read Uncommitted 后续简称 RU):可以读到未提交的读,基本上不会使用该隔离等级,所以暂时忽略。

  • 读已提交(Read Committed 后续简称 RC):存在幻读问题,对当前读获取的数据加记录锁

  • 可重复读(Repeatable Read 后续简称 RR):不存在幻读问题,对当前读获取的数据加记录锁,同时对涉及的范围加间隙锁,防止新的数据插入,导致幻读。

  • 序列化(Serializable):从 MVCC 并发控制退化到基于锁的并发控制,不存在快照读,都是当前读,并发效率急剧下降,不建议使用。

这里说明一下,RC 总是读取记录的最新版本,而 RR 是读取该记录事务开始时的那个版本,虽然这两种读取的版本不同,但是都是快照数据,并不会被写操作阻塞,所以这种读操作称为 快照读(Snapshot Read)

MySQL 还提供了另一种读取方式叫当前读(Current Read),它读的不再是数据的快照版本,而是数据的最新版本,并会对数据加锁,根据语句和加锁的不同,又分成三种情况:

  • SELECT ... LOCK IN SHARE MODE:加共享(S)锁

  • SELECT ... FOR UPDATE:加排他(X)锁

  • INSERT / UPDATE / DELETE:加排他(X)锁

当前读在 RR 和 RC 两种隔离级别下的实现也是不一样的:RC 只加记录锁,RR 除了加记录锁,还会加间隙锁,用于解决幻读问题

不同 SQL 语句对加锁的影响

不同的 SQL 语句当然会加不同的锁,总结起来主要分为五种情况:

  • SELECT ... 语句正常情况下为快照读,不加锁;

  • SELECT ... LOCK IN SHARE MODE 语句为当前读,加 S 锁;

  • SELECT ... FOR UPDATE 语句为当前读,加 X 锁;

  • 常见的 DML 语句(如 INSERT、DELETE、UPDATE)为当前读,加 X 锁;

  • 常见的 DDL 语句(如 ALTER、CREATE 等)加表级锁,且这些语句为隐式提交,不能回滚。

其中,当前读的 SQL 语句的 where 从句的不同也会影响加锁,包括是否使用索引,索引是否是唯一索引等等。

当前数据对加锁的影响

SQL 语句执行时数据库中的数据也会对加锁产生影响。

比如一条最简单的根据主键进行更新的 SQL 语句,如果主键存在,则只需要对其加记录锁,如果不存在,则需要在加间隙锁。

至于其他非唯一性索引更新或者插入时的加锁也都不同程度的受到现存数据的影响,后续我们会一一说明。

具体场景分析

具体 SQL 场景分析主要借鉴何登成前辈的《MySQL 加锁处理分析》文章和 aneasystone 的系列文章,在他们的基础上进行了总结和整理。

我们使用下面这张 book 表作为实例,其中 id 为主键,ISBN(书号)为二级唯一索引,Author(作者)为二级非唯一索引,score(评分)无索引。

MySQL中有哪些加锁场景

UPDATE 语句加锁分析

下面,我们先来分析 UPDATE 相关 SQL 在使用较为简单 where 从句情况下加锁情况。其中的分析原则也适用于 UPDATE,DELETE 和 SELECT ... FOR UPDATE等当前读的语句。

聚簇索引,查询命中

聚簇索引就是 InnoDB 存储引擎下的主键索引,具体可参考《MySQL索引》。

下图展示了使用 UPDATE book SET score = 9.2 WHERE ID = 10 语句命中的情况下在 RC 和 RR 隔离等级下的加锁,两种隔离等级下没有任何区别,都是对 ID = 10 这个索引加排他记录锁。

MySQL中有哪些加锁场景

聚簇索引,查询未命中

下图展示了 UPDATE book SET score = 9.2 WHERE ID = 16 语句未命中时 RR 隔离级别下的加锁情况。

在 RC 隔离等级下,不需要加锁;而在 RR 隔离级别会在 ID = 16 前后两个索引之间加上间隙锁。

MySQL中有哪些加锁场景

值得注意的是,间隙锁和间隙锁之间是互不冲突的,间隙锁唯一的作用就是为了防止其他事务的插入新行,导致幻读,所以加间隙 S 锁和加间隙 X 锁没有任何区别。

二级唯一索引,查询命中

下图展示了 UPDATE book SET score = 9.2 WHERE ISBN = 'N0003' 在 RC 和 RR 隔离等级下命中时的加锁情况。

在 InnoDB 存储引擎中,二级索引的叶子节点保存着主键索引的值,然后再拿主键索引去获取真正的数据行,所以在这种情况下,二级索引和主键索引都会加排他记录锁。

MySQL中有哪些加锁场景

二级唯一索引,查询未命中

下图展示了 UPDATE book SET score = 9.2 WHERE ISBN = 'N0008' 语句在 RR 隔离等级下未命中时的加锁情况,RC 隔离等级下该语句未命中不会加锁。

因为  N0008 大于 N0007,所以要锁住 (N0007,正无穷)这段区间,而 InnoDB 的索引一般都使用 Suprenum Record 和  Infimum Record 来分别表示记录的上下边界。Infimum 是比该页中任何记录都要小的值,而 Supremum  比该页中最大的记录值还要大,这两条记录在创建页的时候就有了,并且不会删除。

所以,在 N0007 和 Suprenum Record 之间加了间隙锁。

MySQL中有哪些加锁场景

为什么不在主键上也加 GAP 锁呢?欢迎留言说出你的想法。

二级非唯一索引,查询命中

下图展示了 UPDATE book SET score = 9.2 WHERE Author = 'Tom' 语句在 RC 隔离等级下命中时的加锁情况。

我们可以看到,在 RC 等级下,二级唯一索引和二级非唯一索引的加锁情况是一致的,都是在涉及的二级索引和对应的主键索引上加上排他记录锁。

MySQL中有哪些加锁场景

但是在 RR 隔离等级下,加锁的情况产生了变化,它不仅对涉及的二级索引和主键索引加了排他记录锁,还在非唯一二级索引上加了三个间隙锁,锁住了两个 Tom 索引值相关的三个范围。

那为什么唯一索引不需要加间隙锁呢?间隙锁的作用是为了解决幻读,防止其他事务插入相同索引值的记录,而唯一索引和主键约束都已经保证了该索引值肯定只有一条记录,所以无需加间隙锁。

MySQL中有哪些加锁场景

需要注意的是,上图虽然画着 4 个记录锁,三个间隙锁,但是实际上间隙锁和它右侧的记录锁会合并成 Next-Key 锁。

所以实际情况有两个 Next-Key 锁,一个间隙锁(Tom60,正无穷)和两个记录锁。

二级非唯一索引,查询未命中

下图展示了 UPDATE book SET score = 9.2 WHERE Author = 'Sarah' 在 RR 隔离等级下未命中的加锁情况,它会在二级索引 Rose 和 Tom 之间加间隙锁。而 RC 隔离等级下不需要加锁。

MySQL中有哪些加锁场景

无索引

当 Where 从句的条件并不使用索引时,则会对全表进行扫描,在 RC 隔离等级下对所有的数据加排他记录锁。在RR 隔离等级下,除了给记录加锁,还会对记录和记录之间加间隙锁。和上边一样,间隙锁会和左侧的记录锁合并成 Next-Key 锁。

下图就是 UPDATE book SET score = 9.2 WHERE score = 22 语句在两种隔离等级下的加锁情况。

MySQL中有哪些加锁场景

聚簇索引,范围查询

上面介绍的场景都是 where 从句的等值查询,而范围查询的加锁又是怎么样的呢?我们慢慢来看。

下图是 UPDATE book SET score = 9.2 WHERE ID <= 25 在 RC 和 RR 隔离等级下的加锁情况。

RC 场景下与等值查询类似,只会在涉及的 ID = 10,ID = 18 和 ID = 25 索引上加排他记录锁。

MySQL中有哪些加锁场景

而在 RR 隔离等级下则有所不同,它会加上间隙锁,和对应的记录锁合并称为 Next-Key 锁。除此之外,它还会在(25, 30] 上分别加 Next-Key 锁。这一点是十分特殊的,具体原因还需要再探究。

二级索引,范围查询

下图展示了 UPDATE book SET ISBN = N0001 WHERE score <= 7.9 在 RR 级别下的加锁情况。

MySQL中有哪些加锁场景

修改索引值

UPDATE 语句修改索引值的情况可以分开分析,首先 Where 从句的加锁分析如上文所述,多了一步 Set 部分的加锁。

下图展示了 UPDATE book SET Author = 'John' WHERE ID = 10 在 RC 和 RR 隔离等级下的加锁情况。除了在主键 ID 上进行加锁,还会对二级索引上的 Bob(就值) 和 John(新值) 上进行加锁。

MySQL中有哪些加锁场景

DELETE 语句加锁分析

一般来说,DELETE 的加锁和 SELECT FOR UPDATE 或 UPDATE 并没有太大的差异。

因为,在  MySQL 数据库中,执行 DELETE 语句其实并没有直接删除记录,而是在记录上打上一个删除标记,然后通过后台的一个叫做 purge  的线程来清理。从这一点来看,DELETE 和 UPDATE 确实是非常相像。事实上,DELETE 和 UPDATE 的加锁也几乎是一样的。

INSERT 语句加锁分析

接下来,我们来看一下 Insert 语句的加锁情况。

Insert 语句在两种情况下会加锁:

  • 为了防止幻读,如果记录之间加有间隙锁,此时不能 Insert;

  • 如果 Insert 的记录和已有记录造成唯一键冲突,此时不能 Insert;

除了上述情况,Insert 语句的锁都是隐式锁。隐式锁是 InnoDB 实现的一种延迟加锁的机制来减少加锁的数量。

隐式锁的特点是只有在可能发生冲突时才加锁,减少了锁的数量。另外,隐式锁是针对被修改的 B+Tree 记录,因此都是记录类型的锁,不可能是间隙锁或 Next-Key 类型。

具体 Insert 语句的加锁流程如下:

  • 首先对插入的间隙加插入意向锁(Insert Intension Locks) 如果该间隙已被加上了间隙锁或 Next-Key 锁,则加锁失败进入等待; 如果没有,则加锁成功,表示可以插入;

  • 然后判断插入记录是否有唯一键,如果有,则进行唯一性约束检查  如果不存在相同键值,则完成插入 如果存在相同键值,则判断该键值是否加锁 如果没有锁, 判断该记录是否被标记为删除  如果标记为删除,说明事务已经提交,还没来得及 purge,这时加 S 锁等待; 如果没有标记删除,则报 duplicate key 错误;  如果有锁,说明该记录正在处理(新增、删除或更新),且事务还未提交,加 S 锁等待;

  • 插入记录并对记录加 X 记录锁;

看完上述内容是否对您有帮助呢?如果还想对相关知识有进一步的了解或阅读更多相关文章,请关注亿速云行业资讯频道,感谢您对亿速云的支持。

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

MySQL中有哪些加锁场景

下载Word文档到电脑,方便收藏和打印~

下载Word文档

猜你喜欢

浅谈MySQL 有哪些死锁场景

目录1 环境准备2 同一张表下的死锁3 不同表下的死锁4 间隙锁下的死锁5 INSERT 语句的死锁首先一起来复习一下死锁的概念:死锁是指两个或者多个事务在同一资源上相互占用,并请求锁定对方占用的资源,从而导致恶性循环的现象。下面我们通过
2022-08-11

mysql乐观锁使用的场景有哪些

乐观锁通常在以下场景中应用:并发读取:多个用户可以同时读取同一数据,不会产生冲突。并发更新:多个用户可以同时读取同一数据,但只有一个用户可以成功更新数据。数据量大:当数据量较大时,使用乐观锁可以减少数据库的锁定时间,提高性能。业务场景
mysql乐观锁使用的场景有哪些
2024-03-15

mysql悲观锁的使用场景有哪些

高并发更新操作:当有多个用户同时对同一行数据进行更新操作时,为了避免数据冲突和丢失,可以使用悲观锁来保证数据的一致性。数据库事务:在数据库事务中,为了确保事务的完整性和一致性,可以使用悲观锁来锁定需要操作的数据,防止其他事务对数据进行并发修
mysql悲观锁的使用场景有哪些
2024-05-08

在Java并发场景中有哪些锁

这篇文章主要介绍了在Java并发场景中有哪些锁,具有一定借鉴价值,感兴趣的朋友可以参考下,希望大家阅读完这篇文章之后大有收获,下面让小编带着大家一起了解一下。01.乐观锁 vs 悲观锁乐观锁与悲观锁是一种广义上的概念,体现了看待线程同步的不
2023-06-16

redis锁的应用场景有哪些

Redis锁的应用场景有以下几个:1. 分布式锁:在分布式系统中,多个节点需要共享某个资源时,使用Redis锁可以保证在同一时间只有一个节点能够访问该资源,以避免并发冲突。2. 并发控制:当多个线程或进程需要同时对某个共享资源进行操作时,使
2023-08-12

mysql中insert的应用场景有哪些

在MySQL中,INSERT语句用于将新的数据行插入到表中。以下是一些INSERT语句的应用场景:添加新数据:最常见的用法是向数据库表中插入新的数据行。这可以通过指定要插入的列及其对应的值来实现。复制数据:可以使用INSERT INTO
mysql中insert的应用场景有哪些
2024-03-06

mysql中div的应用场景有哪些

在MySQL中,DIV函数通常用于做除法运算,将一个数字除以另一个数字,并返回结果。应用场景包括:求商:可以使用DIV函数进行整数除法运算,返回商的整数部分。SELECT 10 DIV 3; -- 结果为3求余数:可以结合MOD函数一起使
mysql中div的应用场景有哪些
2024-05-10

java读写锁的使用场景有哪些

当多个线程需要读取共享数据,但只有少数线程需要写入数据时,使用读写锁可以提高并发性能。当对共享数据的读操作比写操作频繁时,使用读写锁可以减少写操作的竞争,提高系统性能。当需要保证对数据的读写操作是线程安全的时候,可以使用读写锁来控制并发访问
java读写锁的使用场景有哪些
2024-04-03

MySQL锁机制&&PHP锁机制,应用在哪些场景中?

正文内容 模拟准备--如何模拟高并发访问一个脚本:apache安装文件的bin/ab.exe可以模拟并发量 -c 模拟多少并发量 -n 一共请求多少次 http://请求的脚本 例如:cmd: apache安装路径/bin/ab.exe -c 10 -n
MySQL锁机制&&PHP锁机制,应用在哪些场景中?
2016-06-02

mysql的mvcc应用场景有哪些

MySQL的MVCC(多版本并发控制)主要应用于以下场景:事务并发控制:MVCC能够保证事务的并发执行,使得多个事务可以同时读取和更新数据,而不会发生数据不一致的情况。高并发读写:MVCC能够有效地减少锁的使用,提高数据库的并发读写性能。
mysql的mvcc应用场景有哪些
2024-04-24

java可重入锁的使用场景有哪些

可重入锁是指同一个线程可以多次获得同一把锁,在释放锁之前需要释放相同次数的锁。可重入锁的使用场景包括:1. 递归函数:当一个递归函数需要获取锁来保护共享资源时,可重入锁可以允许递归函数多次获取同一把锁。2. 锁的嵌套:当一个方法A获得了锁之
2023-09-11

mysql中date_format函数的应用场景有哪些

将日期格式化为特定的字符串格式,如将"2021-10-15"格式化为"15/10/2021"。在SELECT语句中,将日期字段按照特定的格式显示,方便查看和分析。将日期字段进行比较或排序,需要先将日期格式化为统一的格式。将日期字段进行
mysql中date_format函数的应用场景有哪些
2024-04-09

vuex中有哪些使用场景

这篇文章将为大家详细讲解有关vuex中有哪些使用场景,小编觉得挺实用的,因此分享给大家做个参考,希望大家阅读完这篇文章后可以有所收获。vue是什么Vue是一套用于构建用户界面的渐进式JavaScript框架,Vue与其它大型框架的区别是,使
2023-06-14

MySQL中cast函数的应用场景有哪些

类型转换:cast函数用于将一个数据类型转换为另一个数据类型。例如,将一个字符串转换为整数、将一个浮点数转换为整数等。数据格式化:cast函数可以用于格式化日期、时间或数字等数据类型。例如,将日期格式化为特定的格式、将数字格式化为特定的精度
MySQL中cast函数的应用场景有哪些
2024-04-09

编程热搜

目录