京东6.18大促主会场领京享红包更优惠

 找回密码
 立即注册

QQ登录

只需一步,快速开始

深入理解MySQL的行级锁

2024-11-2 22:42| 发布者: 8b79| 查看: 43| 评论: 0

摘要: 目录☃️概述☃️行级锁先容☃️行锁❄️❄️先容❄️❄️演示☃️间隙锁&临键锁☃️概述 锁是计算机协调多个进程或线程并发访问某一资源的机制。在数据库中,除传统的计算资源(CPU、RAM、I/O)的争用以外,数据也
目录

☃️概述

锁是计算机协调多个进程或线程并发访问某一资源的机制。在数据库中,除传统的计算资源(CPU、RAM、I/O)的争用以外,数据也是一种供许多用户共享的资源。

怎样包管数据并发访问的同等性、有效性是全部数据库必须解决的一个问题,锁冲突也是影响数据库并发访问性能的一个重要因素。从这个角度来说,锁对数据库而言显得尤其重要,也更加复杂。

MySQL中的锁,按照锁的粒度分,分为以下三类:

  • 全局锁:锁定命据库中的全部表。
  • 表级锁:每次操作锁住某一张表。
  • 行级锁:每次操作锁住对应的行数据。

☃️行级锁先容

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

对于行级锁,重要分为以下三类:

  • 行锁(Record Lock):锁定单个行记录的锁,防止其他事件对此行进行update和delete。在RC、RR隔离级别下都支持。

  • 间隙锁(Gap Lock):锁定索引记录间隙(不含该记录),确保索引记录间隙不变,防止其他事件在这个间隙进行insert,产生幻读。在RR隔离级别下支持。

在这里插入图片描述

  • 临键锁(Next-Key Lock):行锁和间隙锁组合,同时锁住数据,并锁住数据前面的间隙Gap。在RR隔离级别下支持。

在这里插入图片描述

☃️行锁

❄️❄️先容

[code]InnoDB[/code] 实现了以下两种范例的行锁:

  • 共享锁(S):答应一个事件去读一行,阻止其他事件获得相同数据集的排它锁。
  •  排他锁(X):答应获取排他锁的事件更新数据,阻止其他事件获得相同数据集的共享锁和排他锁。

两种行锁的兼容情况如下:

锁范例SX
S兼容冲突
X冲突兼容

常见的SQL语句,在实行时,所加的行锁如下:

在这里插入图片描述

❄️❄️演示

默认情况下,[code]InnoDB[/code] 在 [code]REPEATABLE READ[/code] 事件隔离级别运行,[code]InnoDB[/code] 使用 [code]next-key[/code] 锁进行搜索和索引扫描,以防止幻读。

● 针对唯一索引进行检索时,对已存在的记录进行等值匹配时,将会主动优化为行锁。
● InnoDB的行锁是针对于索引加的锁,不通过索引条件检索数据,那么InnoDB将对表中的全部记录加锁,此时 就会升级为表锁。

可以通过以下SQL,检察意向锁及行锁的加锁情况:

A. 平常的select语句,实行时,不会加锁。

B. select…lock in share mode,加共享锁,共享锁与共享锁之间兼容。

共享锁与排他锁之间互斥。

客户端一获取的是id为1这行的共享锁,客户端二是可以获取id为3这行的排它锁的,因为不是同一行数据。 而假如客户端二想获取id为1这行的排他锁,会处于壅闭状态,以为共享锁与排他锁之间互斥。

C. 排它锁与排他锁之间互斥

当客户端一,实行update语句,会为id为1的记录加排他锁; 客户端二,假如也实行update语句更新id为1的数据,也要为id为1的数据加排他锁,但是客户端二会处于壅闭状态,因为排他锁之间是互斥的。 直到客户端一,把事件提交了,才会把这一行的行锁开释,此时客户端二,解除壅闭。

D. 无索引行锁升级为表锁
我们在两个客户端中实行如下操作:

在客户端一中,开启事件,并实行update语句,更新name为Lily的数据,也就是id为19的记录 。然后在客户端二中更新id为3的记录,却不能直接实行,会处于壅闭状态,为什么呢?
缘故因由就是因为此时,客户端一,根据name字段进行更新时,name字段是没有索引的,假如没有索引,此时行锁会升级为表锁(因为行锁是对索引项加的锁,而name没有索引)。
接下来,我们再针对name字段创建索引,索引创建之后,再次做一个测试:

此时我们可以看到,客户端一,开启事件,然后依然是根据name进行更新。而客户端二,在更新id为3的数据时,更新成功,并未进入壅闭状态。 这样就阐明,我们根据索引字段进行更新操作,就可以制止行锁升级为表锁的情况。

☃️间隙锁&临键锁

默认情况下,InnoDB在 REPEATABLE READ事件隔离级别运行,InnoDB使用 next-key 锁进行搜索和索引扫描,以防止幻读。

  • 索引上的等值查询(唯一索引),给不存在的记录加锁时, 优化为间隙锁 。
  • 索引上的等值查询(非唯一平常索引),向右遍历时末了一个值不满意查询需求时,next-key lock 退化为间隙锁。
  • 索引上的范围查询(唯一索引)–会访问到不满意条件的第一个值为止。

注意:
间隙锁唯一目标是防止其他事件插入间隙。间隙锁可以共存,一个事件接纳的间隙锁不会阻止另一个事件在同一间隙上接纳间隙锁。

示例演示
A. 索引上的等值查询(唯一索引),给不存在的记录加锁时, 优化为间隙锁 。

在这里插入图片描述

B. 索引上的等值查询(非唯一平常索引),向右遍历时末了一个值不满意查询需求时,next-key lock 退化为间隙锁。

先容分析一下:

我们知道InnoDB的B+树索引,叶子节点是有序的双向链表。 假如,我们要根据这个二级索引查询值为18的数据,并加上共享锁,我们是只锁定18这一行就可以了吗? 并不是,因为黑白唯一索引,这个结构中大概有多个18的存在,以是,在加锁时会继续往后找,找到一个不满意条件的值(当前案例中也就是29)。此时会对18加临键锁,并对29之前的间隙加锁。

C. 索引上的范围查询(唯一索引)–会访问到不满意条件的第一个值为止。

查询的条件为id>=19,并添加共享锁。 此时我们可以根据数据库表中现有的数据,将数据分为三个部分:

[code][19](19,25](25,+∞][/code]

以是数据库数据在加锁是,就是将19加了行锁,25的临键锁(包含25及25之前的间隙),正无穷的临键锁(正无穷及之前的间隙)。

到此这篇关于深入理解MySQL的行级锁的文章就先容到这了,更多相关MySQL 行级锁内容请搜索脚本之家以前的文章或继续欣赏下面的相关文章希望大家以后多多支持脚本之家!


来源:https://www.jb51.net/database/327133kow.htm
免责声明:如果侵犯了您的权益,请联系站长,我们会及时删除侵权内容,谢谢合作!
关闭

站长推荐上一条 /6 下一条

QQ|手机版|小黑屋|梦想之都-俊月星空 ( 粤ICP备18056059号 )|网站地图

GMT+8, 2025-7-15 22:30 , Processed in 0.036180 second(s), 18 queries .

Powered by Mxzdjyxk! X3.5

© 2001-2025 Discuz! Team.

返回顶部