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的个数恒为奇数或偶数…

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

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

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

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

版本控制工具-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…

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

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

国产化操作系统杂谈

目录 操作系统国产化背景国产化操作系统名录优秀操作系统介绍1.深度Linux(deepin)2.FydeOS3.AliOS(openAliOS)4.openEuler5.红旗Linux6. startOS 总结 操作系统国产化背景 官方的说法是为了打破长期以来国外对中国的操作系统的垄…

OGG几何内核开发-复杂装配模型读取、显示、分析

OGG几何内核读取STEP模型文件的API有STEPCAFControl_Reader、STEPControl_Reader。 STEPCAFControl_Reader使用很复杂,但可以展示装配树,有利于模型的详细分析。 本文演示了《插件化算法研究平台V2》的OCC几何模型插件的部分功能:显示装配树…

四川汇聚荣科技有限公司靠谱吗?

在如今这个信息爆炸的时代,了解一家公司是否靠谱对于消费者和合作伙伴来说至关重要。四川汇聚荣科技有限公司作为一家位于中国西部地区的企业,自然也受到了人们的关注。那么,这家公司究竟如何呢?接下来,我们将从多个角度进行深入…

Python应用开发——30天学习Streamlit Python包进行APP的构建(8)

st.table 显示静态表格。 这与 st.dataframe 的不同之处在于,这里的表格是静态的:其全部内容直接显示在页面上。 Function signature[source]st.table(data=None) Parametersdata (pandas.DataFrame, pandas.Styler, pyarrow.Table, numpy.ndarray, pyspark.sql.DataFrame,…

Mysql数据库约束的概述 , 逐渐约束 , 主键自增 , 表关系的概念和外键 ,多表关系约束介绍和使用

约束和表设计 1、DQL查询语句-limit语句(掌握) 目标 能够掌握limit语句的使用 讲解 作用: LIMIT是限制的意思,所以LIMIT的作用就是限制查询记录的条数。 LIMIT语句格式: select * from 表名 limit offset, row_count; mysql中limit的用法&#…

面试题4:POST 比 GET 安全?

不是。HTTP就没有加密功能。 我们知道 GET一般将参数放到URL的查询字符串中,如果是实现登录页面,我们的用户名和密码就直接显示到浏览器的地址栏中了,此时就会轻易的被他人获取账号密码,很不安全。而POST会把参数放到 body 里&am…

Flutter框架高阶——Window应用程序设置窗体窗口背景完全透明

文章目录 1.修改 main.cpp1)C 与 Win32 API2)EnableTransparency()3)中文注释 2.编写 Flutter 代码1)bitsdojo_window2)window_manager3)区别对比4)同时使用(1)设置初始化…

winRar去广告记录

效果:双击winRar.exe,不会弹出广告窗口,但会弹出使用时间许可警告,效果不是很完美。 工具:everything.exe(非必须)、sublime text(非必须)、spyxx.exe(非必须…

【Python】已解决:pymssql引发的MSSQLDatabaseException错误

文章目录 一、分析问题背景二、可能出错的原因三、错误代码示例四、正确代码示例五、注意事项 已解决:pymssql引发的MSSQLDatabaseException错误 一、分析问题背景 在Python中使用pymssql库与Microsoft SQL Server数据库交互时,有时会遇到pymssql._mss…