MySQL锁详解

目录

前言

MySQL锁

共享锁和排他锁 - Shared and Exclusive Locks

意向锁 - Intention Locks

索引记录锁 - Record Locks 

间隙锁 - Gap Locks

临键锁 - Next-Key Locks

插入意向锁 - Insert Intention Locks

AUTO-INC Locks

死锁

死锁产生条件

InnoDB对死锁的检测


前言

    在高并发场景下数据库性能往往是尤为重要的,通常会采用一些缓存中间件来解决数据库并发问题。这篇文章将介绍数据库层面一些与并发有关的知识。

MySQL锁

1)锁类型

    锁类型依赖于存储引擎,在InnoDB存储引擎中按照锁的粒度分为,行级锁 RECORD ,和表级锁TABLE

    行级锁:是对表中的某些具体的数据行进行加锁

    表级锁:是对整个数据表进行加锁

2)锁模式

    锁模式,⽤来描述如何请求(申请)锁,分为共享锁(S)、独占锁(X)、意向共享锁(IS)、意向独占锁(IX)、记录锁(精准行锁)、间隙锁、Next-Key锁(临键锁)、AUTO-INC锁、空间索引的谓词锁等。

共享锁和排他锁 - Shared and Exclusive Locks

    InnoDB实现了标准的行级锁,分为两种分别是共享锁(S锁)和独占锁(X锁),独占锁也称为排他锁。

    共享锁(S锁):允许持有该锁的事务读取表中的一行记录,同时允许其他事务在锁定行上加另⼀个共享锁并读取被锁定的对象,但不能对其进行写操作,也就是加排他锁需要阻塞等待。

    独占锁(X锁):允许持有该锁的事务对数据行进行更新或删除,同时不论其他事务对锁定行进行读取或修改都不允许对锁定行进行加锁,也就是加共享锁和排他锁都会阻塞。

示例:

  • 如果事务T1持有R行上的共享锁(S),那么事务T2请求R行上的锁时会有如下处理:
    • T2请求S锁会立即被授予,此时T1和T2都对R行持有S锁
    • T2请求X锁不能立即被授予,阻塞到T1释放持有的锁
  • 如果事务T1持有R行上的独占锁(X),那么T2请求R行上的任意类型锁都不能立即被授予,事务T2必须等待事务T1释放R行上的锁

 读锁是共享锁的一种实现,写锁是排他锁的一种实现

意向锁 - Intention Locks 

 InnoDB支持多粒度锁,允许行锁和表锁共存

  • InnoDB使⽤意向锁实现多粒度级别的锁,意向锁是表级别的锁,它并不是真正意义上的加锁,而只是在 data_locks 中记录事务以后要对表中的哪一行加哪种类型的锁(共享锁或排他锁),意向锁分为两种:
    • 意向共享锁(IS):表示事务打算对表中的单个行设置共享锁。
    • 意向排他锁(IX):表示事务打算对表中的单个行设置排他锁。
  • 在获取意向锁时有如下协议:
    • 在事务获得表中某一行的共享锁(S)之前,它必须⾸先获得该表上的IS锁或更强的锁。
    • 在事务获得表中某一行的排他锁(X)之前,它必须⾸先获得该表上的IX锁。 
  • 意向锁可以提⾼加锁的性能,在真正加锁之前不需要遍历表中的行是否加锁,只需要查看⼀下表中的意向锁即可
  • 在请求锁的过程中,如果将要请求的锁与现有锁兼容,则将锁授予请求的事务,如果与现有锁冲突,则不会授予;事务将阻塞等待,直到冲突的锁被释放
  • 除了全表锁定请求之外,意向锁不会阻⽌任何锁请求;意向锁的主要⽬的是表示事务正在锁定某行或者正在意图锁定某行

索引记录锁 - Record Locks 

 索引记录锁称为精准行锁,顾名思意是指索引记录上的锁,如下SQL锁住的是指定的一行

# 在MySQL8中使用for update进行加排他锁
select * from user where id = 1 for update;

    索引记录锁总是锁定索引行,在表没有定义索引的情况下,InnoDB创建一个隐藏的聚集索引,并使用该索引进行记录锁定,当使用索引进行查找时,锁定的只是满足条件的行。

间隙锁 - Gap Locks

    间隙锁锁定的是索引记录之间的间隙,或者第一个索引记录之前,再或者最后一个索引记录之后的间隙。如图所示位置,根据不同的查询条件都可能会加间隙锁。

    例如有如下SQL,锁定的是ID (10, 20)之间的间隙,注意不包括10和20的行,目的是防止其他事务将ID值为15的列插入到列 user  表中(无论是否已经存在要插入的数据列),因为指定范围值之间的间隙被锁定了。

select * from user where id between 10 and 20 for update
  • 间隙可以跨越单个或多个索引值
  • 对于使⽤唯一索引查询到的唯一行,不使用间隙锁,如下语句,id列有唯一的索引,只对id值为100的行使用索引记录锁:
    • select * from user where id = 1;
  • 如果id没有被索引,或者是一个非唯一的索引,以上语句将锁定对应记录前面的间隙
  • 不同事务的间隙锁可以共存,一个事务的间隙锁不会阻止另一个事务在相同的间隙上使用间隙锁;共享间隙锁和独占间隙锁之间没有区别
  • 当事务隔离级别设置为 READ COMMITTED 时间隙锁会被禁用,对于搜索和索引扫描不再使用间隙锁定

临键锁 - Next-Key Locks

    Next-Key锁是索引记录锁和间隙锁的组合。它使用索引记录锁锁定具体行,再使用间隙锁锁住该行前面的间隙,等于就是一开一闭的区间。

  • InnoDB搜索或扫描⼀个表的索引时,执行行级锁策略,具体方式是:在扫描过程中遇到的索引记录上设置共享锁或排他锁,因此,行级锁策略实际上应用的是索引记录锁。索引记录上的 next-key 锁也会影响该索引记录之前的"间隙",也就是说, next-key 锁是索引记录锁加上索引记录前面的间隙锁。如果一个会话对索引中的一条记录R具有共享锁或排他锁,则另一个会话不能在索引记录R之前的空白中插⼊新的索引记录行
  • 假设索引包含值10、11、13和20,这些索引可能的 next-key 锁覆盖以下区间,其中圆括号表示不包含区间端点,方括号表示包含端点:
    • (负无穷,10 ],(10,11 ],(11,13 ],(13, 20 ],(20,正无穷 ]
  • 默认情况下, REPEATABLE READ 事务隔离级别开启 next-key 锁并进行搜索和索引扫描,可以防止幻象行,从而解决幻读问题

插入意向锁 - Insert Intention Locks

    插⼊意向锁是⼀个特殊的间隙锁,在向索引记录之前的间隙进行insert操作插⼊数据时使用,如果多个事务向相同索引间隙中不同位置插入记录,则不需要彼此等待。假设已经存在值为10和20的索引记录,两个事务分别尝试插入索引值为15和16的行,在获得插入行上的排他锁之前,每个事务都⽤插⼊意向锁锁住10到20之间的间隙,但不会相互阻塞,因为他们所操作的行并不冲突。

    插入意向锁只是标记某个事务要对某行进行插入数据,不同事务之间获取插入意向锁不会产生锁竞争。

示例:

    事务A要对事务B使用排他锁锁定的区间进行插入数据,事务A首先会标记插入意向锁,然后获取排他锁阻塞等待,直到事务B事务结束释放排他锁,这个时候事务A就可获取到锁,进行插入数据。

AUTO-INC Locks

    AUTO-INC锁也叫自增锁是一个表级锁,服务于配置了 AUTO_INCREMENT 自增列的表。在插入数据时会在表上加自增锁,并生成自增值,同时阻塞其他的事务操作,以保证值的唯一性。需要注意的是,当一个事务执行新增操作已生成自增值,但是事务回滚了,申请到的主键值不会回退,这意味着在表中会出现自增值不连续的情况。

死锁

    由于每个事务都持有另一个事务所需的锁,导致事务无法继续进行的情况称为死锁。以下图为例,两个事务都不会主动释放自己持有的锁,并且都在等待对方持有的资源变得可用。

死锁产生条件

1)互斥访问:如果线程1获取到了锁A,那么线程2就不能同时得到锁A

2)不可抢占:获取到锁的线程,只能自己主动释放锁,别的线程不能从他的手中抢占锁

3)保持与请求:线程1已经获得了锁A,还要在这个基础上再去获了锁B

4)循环等待:线程1等待线程2释放锁,线程2也等待线程1释放锁,死锁发⽣时系统中⼀定有由两个或两个以上的线程组成的⼀条环路,该环路中的每个线程都在等待着下一个进程释放锁

    以上四条是造成死锁的必要条件,必须同时满足,所以如果想要打破死锁,可以破坏以上四个条件之一,最常见的方式就是打破循环等待。

InnoDB对死锁的检测

1)InnoDB在运行时会对死锁进行检测,当死锁检测启用时(默认),InnoDB自动检测事务死锁,并回滚一个或多个事务来打破死锁。InnoDB尝试选择小事务进行回滚,其中事务的大小由插入、更新或删除的行数决定。

2)如果系统变量 innodb_table_locks = 1  (默认) 和 autocommit = 0 ,InnoDB可以检测到表级锁和行级锁级别发生的死锁;否则,无法检测到由 lock TABLES 语句设置的表锁或由非InnoDB存储引擎设置的锁,对于无法检测到的死锁,可以通过设置系统变量innodb_lock_wait_timeout 的值来指定锁的超时时间来解决死锁问题。

3)当超过 200 个事务等待锁资源或等待的锁个数超过 1,000,000 个时也会被视为死锁,并尝试将等待列表的事务回滚。

4)在高并发系统中,多个线程等待相同的锁时,死锁检测可能会导致性能降性变慢,此时禁用死锁检测并依赖 innodb_lock_wait_timeout 设置进行事务回滚可能性能更高。可以通过设置系统变量 innodb_deadlock_detect[={OFF|ON}] 禁用死锁检测。

以下为常用查看MySQL死锁信息命令

# 查看当前服务器发生死锁次数
SELECT `count` FROM INFORMATION_SCHEMA.INNODB_METRICS WHERE NAME="lock_deadlocks";
# 查询死锁和事务相关信息;查看 LATEST DETECTED DEADLOCK 节点的内容
SHOW ENGINE INNODB STATUS
# 查询错误日志路径
SELECT @@log_error;
# 查看锁信息    
SELECT ENGINE_TRANSACTION_ID as Trx_Id, OBJECT_NAME as `Table`, INDEX_NAME as `Index`, LOCK_DATA as Data,LOCK_MODE as Mode, LOCK_STATUS as Status, LOCK_TYPE as TypeFROM performance_schema.data_locks;
# 查看锁等待信息
SELECT REQUESTING_ENGINE_LOCK_ID as Req_Lock_Id, REQUESTING_ENGINE_TRANSACTION_ID as Req_Trx_Id, BLOCKING_ENGINE_LOCK_ID as Blk_Lock_Id, BLOCKING_ENGINE_TRANSACTION_ID as Blk_Trx_IdFROM performance_schema.data_lock_waits
# 查看锁信息
SELECT ENGINE_LOCK_ID as Lock_Id, ENGINE_TRANSACTION_ID as Trx_id, OBJECT_NAME as `Table`, INDEX_NAME as `Index`, LOCK_DATA as Data,LOCK_MODE as Mode, LOCK_STATUS as Status, LOCK_TYPE as Type FROM performance_schema.data_locks;

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/bicheng/32661.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

海康威视-下载的录像视频浏览器播放问题

目录 1、播放异常比对 2、视频编码检查 2.1、正常视频解析 2.2、海康视频解析 2.3、比对工具 3、转码 3.1、maven依赖 3.2、实现代码 4、验证 在前面的文章(海康威视-按时间下载录像文件_海康威视 sdk 下载录像 大小0-CSDN博客)中,通…

计算机网络之奇偶校验码和CRC冗余校验码

今天我们来看看有关于计算机网络的知识——奇偶校验码和CRC冗余校验码,这两种检测编码的方式相信大家在计算机组成原理当中也有所耳闻,所以今天我就来跟大家分享有关他们的知识。 奇偶校验码 奇偶校验码是通过增加冗余位使得码字中1的个数恒为奇数或偶数…

Scikit-learn基础教程:揭开机器学习的神秘面纱

Scikit-learn基础教程:揭开机器学习的神秘面纱 摘要: Scikit-learn是一个开源的Python机器学习库,它提供了一系列易于使用的工具,用于数据挖掘和数据分析。本文将作为一个Scikit-learn基础教程,介绍Scikit-learn的安装…

汽车网络安全 -- 漏洞该如何管理

目录 1.漏洞获取途径汇总 2.CAVD的漏洞管理规则简析 2.1 通用术语简介 2.2 漏洞评分指标 2.3.1 场景参数 2.3.2 威胁参数 2.3.3 影响参数 2.3 漏洞等级判定 ​3.小结 在汽车网络安全的时代背景下,作为一直从事车控类ECU基础软件开发的软件dog,…

MapReduce 实践题:Web 访问日志分析与异常检测

文章目录 作业描述MapReduce 实践题:Web 访问日志分析与异常检测题目背景数据集说明任务要求输入数据示例输出数据示例实现步骤 解题思路1. 数据预处理2. 访问统计3. 异常检测4. 主方法5. 结果输出 作业描述 MapReduce 实践题:Web 访问日志分析与异常检…

mysql启动时遇到:本地计算机上的MySQL服务启动后停止

1.问题重述: 今早启动数据库时发现无法启动,报错:本地计算机 上的 MySQL服务启动后停止。某些服务在未由其他服务或程序使用时将自动停止。 2.解决方案: 1.数据备份: 2.在bin目录下,命令行中输入 mysqld …

【IPython 使用技巧整理】

IPython 使用技巧整理 IPython 是一个交互式 Python 解释器,比标准 Python 解释器提供了更加强大的功能和更友好的使用体验。它为数据科学、机器学习和科学计算提供了强大的工具,是 Python 开发人员不可或缺的工具之一。本文将深入探讨 IPython 的各种使…

Java中的多线程编程实用指南

Java中的多线程编程实用指南 大家好,我是免费搭建查券返利机器人省钱赚佣金就用微赚淘客系统3.0的小编,也是冬天不穿秋裤,天冷也要风度的程序猿!今天我们来探讨一下Java中多线程编程的实用指南。 在当今软件开发的世界中&#x…

python3.9安装pysal库

一、安装geopandas 下载地址:https://download.csdn.net/download/xiangfengl/89330615 然后逐个按顺序安装 1.安装gdal pip install GDAL-3.3.3-cp39-cp39-win_amd64.whl 2.安装Fiona pip install Fiona-1.8.20-cp39-cp39-win_amd64.whl 3.安装shapely pip …

Python面试宝典:云服务概览以及Python在云计算服务中的应用相关面试笔试题(1000加面试笔试题助你轻松捕获大厂Offer)

Python面试宝典:1000加python面试题助你轻松捕获大厂Offer【第二部分:Python高级特性:第二十六章:Python与云计算:第一节:云服务概览以及Python在云计算服务中的应用】 第二十六章:Python与云计算第一节:云服务概览以及Python在云计算服务中的应用1. 云计算服务的开发与…

版本控制工具-git的基本使用

目录 前言一、git简介二、git工作流程三、git基本命令3.1 创建本地仓库3.2 将工作区内容提交到本地仓库3.3 将本地仓库内容推送到远程仓库 前言 本篇文章介绍git的一般工作流程 一、git简介 Git是一个开源的分布式版本控制软件,常用于项目的版本管理 Git是Linux …

196.每日一题:检测大写字母(力扣)

代码解决 class Solution { public:bool detectCapitalUse(string word) {int capitalCount 0;int n word.size();// 统计大写字母的数量for (char c : word) {if (isupper(c)) {capitalCount;}}// 检查是否满足三种情况之一if (capitalCount n) {// 全部字母都是大写return…

等保测评:全面保障信息系统安全的必要举措

等保测评(信息安全等级保护测评)是评估信息系统安全等级的重要过程,旨在确保信息系统能够抵御各种安全威胁,保障信息的机密性、完整性和可用性。以下是一篇关于等保测评的文章,内容清晰、分点表示,并参考了…

2024.06.23 刷题日记

〇、前言 今天重点刷了回溯,以及常见的题目。 46. 全排列 给定一个不含重复数字的数组 nums ,返回其 所有可能的全排列 。你可以 按任意顺序 返回答案。 示例 1: 输入:nums [1,2,3] 输出:[[1,2,3],[1,3,2],[2,1,3…

Node.js是什么(基础篇)

前言 Node.js是一个基于Chrome V8 JavaScript引擎的开源、跨平台JavaScript运行时环境,主要用于开发服务器端应用程序。它的特点是非阻塞I/O模型,使其在处理高并发请求时表现出色。 一、Node JS到底是什么 1、Node JS是什么 Node.js不是一种独立的编程…

C/C++ - 编码规范(USNA版)

[IC210] Resources/C Programming Guide and Tips 所有提交的评分作业(作业、项目、实验、考试)都必须使用本风格指南。本指南的目的不是限制你的编程,而是为你的程序建立统一的风格格式。 * 这将有助于你调试和维护程序。 * 有助于他人&am…

USB - USB在消费领域的应用

Switching in USB Consumer Applications 通用串行总线(USB)已成为满足终端设备之间日益增长的快速数据传输需求的主流接口--例如,在个人电脑和便携式设备(如手机、数码相机和个人媒体播放器)之间下载和上传数据。 The…

IP地址解析省份区域信息

背景 最近工作需要根据IP地址,解析通话所在省份。因此,在网上找了相关方案,作为存档记录下来。 在线接口 不做阐述。因为,一是网上可以很轻松的找到,没有必要多说。二是开发在内网中,多数不会让连接外网…

Go WebSocket入门+千万级别弹幕系统架构设计

Go实现WebSocket(千万级别弹幕系统架构设计) 1 websocket简介(基于HTTP协议的长连接) 使用WebSocket可以轻松的维持服务器端长连接,其次WebSocket是架构在HTTP协议之上的,并且也可以使用HTTPS方式,因此WebSocket是可靠…

QT事件处理系统之一:父子组件之间的事件传播机制

1、一些解释 /*1. 事件传播(非常重要):GUI编程当中, 事件的传播是[基于组件层面]的, 并不是依靠类继承机制!类的继承:QWidget <- QPushButton <- FirstButton <- SecondButton基于组件(父子组件):QWidget <- QPushButton(FirstButton/SecondButton);虽然…