【MySQL】一文带你理清<行级锁>(行锁,间隙锁,临键锁)

2024-08-26 00:52

本文主要是介绍【MySQL】一文带你理清<行级锁>(行锁,间隙锁,临键锁),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

前言

大家好吖,欢迎来到 YY 滴MySQL系列 ,热烈欢迎! 本章主要内容面向接触过C++ Linux的老铁
主要内容含:
在这里插入图片描述

欢迎订阅 YY滴C++专栏!更多干货持续更新!以下是传送门!

  • YY的《C++》专栏
  • YY的《C++11》专栏
  • YY的《Linux》专栏
  • YY的《数据结构》专栏
  • YY的《C语言基础》专栏
  • YY的《初学者易错点》专栏
  • YY的《小小知识点》专栏
  • YY的《单片机期末速过》专栏
  • YY的《C++期末速过》专栏
  • YY的《单片机》专栏
  • YY的《STM32》专栏
  • YY的《数据库》专栏
  • YY的《数据库原理》专栏

目录

  • 【1】【行锁】(共享锁,排他锁)
    • 1.共享锁,排他锁机制介绍
    • 2.不同SQL下,行锁的情况
    • 3.演示行锁
      • 【1】情况1
      • 【2】情况2
  • 【2】【临键锁S】【间隙锁】演示
    • ※【临键锁S】【间隙锁】特性演示目录
    • 1.演示:索引上的等值查询 (唯一索引,例如主键索引)
    • 2.演示:索引上的范围查询(唯一索引)
    • 3.演示:索引上的等值查询(普通索引)——临键锁退化为间隙锁

【1】【行锁】(共享锁,排他锁)

1.共享锁,排他锁机制介绍

InnoDB实现了以下两种类型的行锁:

  1. 共享锁(S): 允许一个事务去读一行,阻止其他事务获得相同数据集的排它锁。 (共享锁之间是兼容的 ,共享锁与排他锁互斥)
  2. 排他锁(X): 允许获取排他锁的事务更新数据,阻止其他事务获得相同数据集的共享锁和排他锁。 (一个数据有了排他锁,就与其他共享锁和排他锁互斥)

在这里插入图片描述

2.不同SQL下,行锁的情况

  • 分成两种,一种是增删改;另一种是查询
    在这里插入图片描述

3.演示行锁

默认情况下,InnODB在 REPEATABLE READ事务隔离级别运行,InnoDB使用 临键锁 进行搜索和索引扫描,以防止幻读。(本次演示)

  1. 针对 唯一索引 进行检索时,对已存在的记录进行等值匹配时,将会 自动优化为行锁

  2. 不通过索引条件检索数据(InnoDB的行锁是针对于索引加的锁),那么InnoDB将对表中的所有记录加锁,此时 就会升级为表锁

可以通过以下SOL,查看意向锁及行锁的加锁情况:

select object schema,object name,index name,lock type,lock mode,lock data from performance schema.data locks;

【1】情况1

演示:

  • 我们查看一张表,发现表的id是 主键索引

在这里插入图片描述

  • 我们加入共享锁

在这里插入图片描述

  • 我们查看行锁的加锁情况: 注:TABLE 为表锁 RECORD为行锁
  • 查看查看意向锁及行锁的加锁情况:
select object schema,object name,index name,lock type,lock mode,lock data from performance schema.data locks;
  • 发现有共享锁S,且无间隙锁 REC_NOT_GAP

在这里插入图片描述

  • 我们在另一客户端再加上共享锁,依旧能执行;因为 (共享锁之间是兼容的 ,共享锁与排他锁互斥)

在这里插入图片描述

【2】情况2

不通过索引条件检索数据(InnoDB的行锁是针对于索引加的锁),那么InnoDB将对表中的所有记录加锁,此时 就会升级为表锁
演示:

  • 有这么一张表,为主键索引
    在这里插入图片描述
  • 我们针对非索引条件检索数据name,进行更新操作
  • 此时行锁就会升级成表锁
    在这里插入图片描述
  • 此时我们再开一个终端,对id=3的数据行进行修改,发现进入阻塞状态
    在这里插入图片描述

【2】【临键锁S】【间隙锁】演示

※【临键锁S】【间隙锁】特性演示目录

下面进行演示:

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

  1. 索引上的等值查询 (唯一索引,例如主键索引) ,给 不存在的记录 加锁时,优化为间隙锁。
  2. 索引上的范围查询(唯一索引)–会访问到不满足条件的第一个值为止。
  3. 索引上的等值查询(普通索引),向右遍历时最后一个值不满足查询需求时,next-key lock 退化为间隙锁。

1.演示:索引上的等值查询 (唯一索引,例如主键索引)

  • 索引上的等值查询 (唯一索引,例如主键索引) ,给 不存在的记录 加锁时,优化为间隙锁。

  • 表中id为主键索引,我们给不存在的id=5加锁,此时就会在3和8之间加入一个 间隙锁
    在这里插入图片描述

  • 查询发现上了间隙锁

  • 查看查看意向锁及行锁的加锁情况:

select object schema,object name,index name,lock type,lock mode,lock data from performance schema.data locks;

在这里插入图片描述

  • 此时我们往(3-8)的间隙里加入数据(id=7),发现进入阻塞状态
    在这里插入图片描述

2.演示:索引上的范围查询(唯一索引)

  • 索引上的范围查询(唯一索引)–会访问到不满足条件的第一个值为止。
  • 我们针对既是主键也是唯一索引id,进行范围查询
    -
  • 查看锁情况
  • 查看查看意向锁及行锁的加锁情况:
select object schema,object name,index name,lock type,lock mode,lock data from performance schema.data locks;
  • 对19加了一个行锁S,REC_NOT_GAP
  • 对25与25之前间隙加了一个临键锁,S
  • 对25之后到正无穷supremum pseu加了临键锁,S
    在这里插入图片描述

3.演示:索引上的等值查询(普通索引)——临键锁退化为间隙锁

  • 索引上的等值查询(普通索引),向右遍历时最后一个值不满足查询需求时, 临键锁 退化为间隙锁 (可理解成多出一个间隙锁)
  • 前置知识: 我们加的行锁是针对索引加的锁,索引是一个B+树的结构,B+树的节点形成的是一个有序的双向链表
  • 现有的记录中有18,因为其不是唯一索引,18之前与之后将来都可能插入字段值为18的记录
    在这里插入图片描述
  • 于是乎16和18之间,18和29之间都会上锁;18和29之间是间隙锁,而16和18之间的临键锁,此时会退化为间隙锁;

我们可以看看下面这个例子:

  • 我们先对age加上普通索引
    在这里插入图片描述
  • 对age=3的记录,加上共享锁
    在这里插入图片描述
  • 我们查询锁的情况 注:S是临键锁
  • 查看查看意向锁及行锁的加锁情况:
select object schema,object name,index name,lock type,lock mode,lock data from performance schema.data locks;

在这里插入图片描述

  • 3,3是 临键锁S ,对应的是锁住3和3之前的部分在这里插入图片描述

  • 7,7是 临键锁S和 间隙锁GAP ,对应的是所著3和7之间的间隙在这里插入图片描述

  • 向右遍历时最后一个值不满足查询需求时, 临键锁 退化为间隙锁 (可理解成多出一个间隙锁)

这篇关于【MySQL】一文带你理清<行级锁>(行锁,间隙锁,临键锁)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1107086

相关文章

MySQL中时区参数time_zone解读

《MySQL中时区参数time_zone解读》MySQL时区参数time_zone用于控制系统函数和字段的DEFAULTCURRENT_TIMESTAMP属性,修改时区可能会影响timestamp类型... 目录前言1.时区参数影响2.如何设置3.字段类型选择总结前言mysql 时区参数 time_zon

Python MySQL如何通过Binlog获取变更记录恢复数据

《PythonMySQL如何通过Binlog获取变更记录恢复数据》本文介绍了如何使用Python和pymysqlreplication库通过MySQL的二进制日志(Binlog)获取数据库的变更记录... 目录python mysql通过Binlog获取变更记录恢复数据1.安装pymysqlreplicat

使用SQL语言查询多个Excel表格的操作方法

《使用SQL语言查询多个Excel表格的操作方法》本文介绍了如何使用SQL语言查询多个Excel表格,通过将所有Excel表格放入一个.xlsx文件中,并使用pandas和pandasql库进行读取和... 目录如何用SQL语言查询多个Excel表格如何使用sql查询excel内容1. 简介2. 实现思路3

一文带你理解Python中import机制与importlib的妙用

《一文带你理解Python中import机制与importlib的妙用》在Python编程的世界里,import语句是开发者最常用的工具之一,它就像一把钥匙,打开了通往各种功能和库的大门,下面就跟随小... 目录一、python import机制概述1.1 import语句的基本用法1.2 模块缓存机制1.

Mysql DATETIME 毫秒坑的解决

《MysqlDATETIME毫秒坑的解决》本文主要介绍了MysqlDATETIME毫秒坑的解决,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着... 今天写代码突发一个诡异的 bug,代码逻辑大概如下。1. 新增退款单记录boolean save = s

mysql-8.0.30压缩包版安装和配置MySQL环境过程

《mysql-8.0.30压缩包版安装和配置MySQL环境过程》该文章介绍了如何在Windows系统中下载、安装和配置MySQL数据库,包括下载地址、解压文件、创建和配置my.ini文件、设置环境变量... 目录压缩包安装配置下载配置环境变量下载和初始化总结压缩包安装配置下载下载地址:https://d

MySQL中的锁和MVCC机制解读

《MySQL中的锁和MVCC机制解读》MySQL事务、锁和MVCC机制是确保数据库操作原子性、一致性和隔离性的关键,事务必须遵循ACID原则,锁的类型包括表级锁、行级锁和意向锁,MVCC通过非锁定读和... 目录mysql的锁和MVCC机制事务的概念与ACID特性锁的类型及其工作机制锁的粒度与性能影响多版本

MYSQL行列转置方式

《MYSQL行列转置方式》本文介绍了如何使用MySQL和Navicat进行列转行操作,首先,创建了一个名为`grade`的表,并插入多条数据,然后,通过修改查询SQL语句,使用`CASE`和`IF`函... 目录mysql行列转置开始列转行之前的准备下面开始步入正题总结MYSQL行列转置环境准备:mysq

MySQL不使用子查询的原因及优化案例

《MySQL不使用子查询的原因及优化案例》对于mysql,不推荐使用子查询,效率太差,执行子查询时,MYSQL需要创建临时表,查询完毕后再删除这些临时表,所以,子查询的速度会受到一定的影响,本文给大家... 目录不推荐使用子查询和JOIN的原因解决方案优化案例案例1:查询所有有库存的商品信息案例2:使用EX

Linux(Centos7)安装Mysql/Redis/MinIO方式

《Linux(Centos7)安装Mysql/Redis/MinIO方式》文章总结:介绍了如何安装MySQL和Redis,以及如何配置它们为开机自启,还详细讲解了如何安装MinIO,包括配置Syste... 目录安装mysql安装Redis安装MinIO总结安装Mysql安装Redis搜索Red