面试官:MySQL 什么时候会出现死锁问题?为什么不推荐使用RR隔离级别?

news2024/11/28 8:28:10

欢迎关注公众号 【11来了】 ,持续 MyBatis 源码系列内容!

在我后台回复 「资料」 可领取编程高频电子书
在我后台回复「面试」可领取硬核面试笔记

文章导读地址:点击查看文章导读!

感谢你的关注!

面试官:MySQL 什么时候会出现死锁问题?为什么不推荐使用RR隔离级别?

MySQL 的死锁问题比较容易在面试中碰到,接下来将会模拟 MySQL 中的死锁现象,通过查看 MySQL 死锁日志来摸清死锁产生原因,并且从死锁产生原因来了解为什么不推荐使用 RR (可重复读)事务隔离级别?

删除数据死锁场景模拟

创建表结构
CREATE TABLE `goods` (
  `id` int(11) NOT NULL,
  `num` int(11) DEFAULT NULL,
  PRIMARY KEY (`id`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci;

INSERT INTO `goods` VALUES (1, 15);
INSERT INTO `goods` VALUES (5, 30);
模拟死锁

可以在 Navicat 客户端打开两个窗口,按照下列顺序执行对应 SQL 语句,即可模拟出死锁现象

事务 A事务 B
start transaction;
delete from goods where id = 1;start transaction;
delete from goods where id = 5;
delete from goods where id = 5;
delete from goods where id = 1; # 死锁

如下,可以用鼠标选中要执行的语句,按照上述顺序执行特定语句:

image-20241004234302807
死锁日志查看

当执行完事务 B 的最后一个语句,Navicat 就会提示死锁

接下来查看死锁的日志,在 MySQL 可以通过 show engine innodb status; 来查看 InnoDB 存储引擎的状态信息,包含了事务、锁等信息

如下图:

image-20241004224021432

接下来将 Status 里的数据粘贴到 Sublime(文本编辑器)中,方便分析日志,如下图,锁的一些信息主要在下方黄色方框内部:

image-20241004224146622

接下来逐个分析,可以看到总共有两个事务,事务 ID 分别为 9511、9512,接下来分别看这两个事务相关的锁信息,先看第一个事务,可以发现第一个事务在等待 id = 5 这一条数据的 X 锁

image-20241004225031403

接下来看一下第二个事务,该事务持有了 id = 5 这条数据的 X 锁,同时在等待 id = 1 这条数据的 X 锁:

image-20241004225221393

由于事务 A 和事务 B 都互相等待对方的锁,因此发生了死锁,通过日志可以看到最后是回滚了第二个事务:

image-20241004225252194
锁日志含义

在使用 show engine innodb status; 查看存储引擎状态时,每一个锁信息都有 4 行记录,这里说一下每条记录的含义:

Record lock, heap no 4 PHYSICAL RECORD: n_fields 4; compact format; info bits 32
 0: len 4; hex 80000005; asc     ;;          # 聚集索引的值
 1: len 6; hex 000000002528; asc     %(;;    # 事务 ID
 2: len 7; hex 78000001bd113d; asc x     =;; # undo 回滚段指针
 3: len 4; hex 8000001e; asc     ;;  	     # 非主键字段值

每个字段的含义在上边已经给出了,第一行记录是 聚集索引的值 ,即 5,那么也就是 id = 5 这一条记录

最后一行记录的值是非主键字段的值,即 1e ,翻译为十进制也就是 30,也就是 id = 5,非主键索引值为 30 这一条记录,如下:

image-20241004225816306

RR 事务隔离级别下造成的死锁

一般在互联网公司中,都不推荐使用 MySQL 的 可重复读 隔离级别,而是更推荐使用 读已提交 隔离级别

原因: 这是因为在 RR 隔离级别下的 间隙锁 容易造成锁等待或死锁,因为 RR 隔离级别需要保证可重复读,MySQL 通过 MVCC + 间隙锁来保证可重复读,如果由于 SQL 语句写的不合适,加的 间隙锁范围过大 ,就会导致在间隙锁范围内无法插入数据,造成锁等待或者死锁

接下来将会模拟 RR 隔离级别下的 间隙锁和插入意向锁冲突 ,从而造成死锁的案例以及对应的日志分析

创建表结构
CREATE TABLE `goods` (
  `id` int(11) NOT NULL,
  `num` int(11) DEFAULT NULL,
  PRIMARY KEY (`id`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci;

INSERT INTO `goods` VALUES (1, 15);
INSERT INTO `goods` VALUES (5, 30);
模拟死锁

如下,同样开启两个 mysql 窗口,按如下顺序执行两个事务,即会发生死锁现象

事务 A事务 B
start transaction;
delete from goods where id = 2;start transaction;
delete from goods where id = 4;
insert into goods values(2, 30)
insert into goods values(4, 30); # 死锁

接下来同样查看死锁日志: show engine innodb status;

image-20241004231959098
死锁日志分析

当执行事务 A 在执行 delete from goods where id = 2 时,由于不存在 id = 2 的数据,因此会在 id 范围(1,5)上添加间隙锁

当事务 A 在执行 insert into goods values(2, 30) 时,如下图,该语句的插入意向锁会与事务 B 持有的范围(1,5)的间隙锁冲突,如下:

image-20241004232216399

对于事务 B 来说,执行 delete from goods where id = 4 时,由于不存在 id = 4 的数据,因此会在 id 范围(1,5)上添加间隙锁

当事务 B 在执行 insert into goods values(4, 30) 时,插入意向锁会与事务 A 持有的范围(1,5)的间隙锁冲突,如下:

image-20241004232637009

因此就会导致死锁现象,上边还是有些绕,最后再简单总结一下,造成死锁的流程为:

  • 事务 A 先执行 delete from goods where id = 2 ,由于并不存在 id = 2 的数据,因此事务 A 会对(1,5)添加间隙锁
  • 事务 B 执行 delete from goods where id = 4 ,同样不存在 id = 4 的数据,因此事务 B 会对(1,5)添加间隙锁
  • 这里事务 A 和事务 B 的间隙锁并不会冲突,因为他是用来防止在间隙中插入新值的,因此会和插入意向锁冲突
  • 之后,事务 A 执行 insert into goods values(2, 30) ,此时会去申请插入意向锁,但是 id = 2 是在范围(1,5)内的,因此该意向锁会和事务 B 持有的(1,5)间隙锁冲突,发生锁等待
  • 之后,事务 B 执行 insert into goods values(4, 30) ,此时就发生了死锁,因为事务 B 申请了 id = 4 的插入意向锁,同样和事务 A 的间隙锁冲突

通过上边两个案例,就可以了解 MySQL 中死锁出现的现象、如何去查看死锁以及为什么不推荐使用 RR 隔离级别

MySQL 中如何查看事务加锁的信息?

需要三步:

  • 设置参数:set global innodb_status_output_locks = ON;
  • 开启事务,并加锁
  • 查看锁信息:show engine innodb status;

接下来演示一下,在 Navicat 打开一个查询窗口,按照顺序执行上边三个步骤:

image-20241005132703335

最后一步会打印出来 innodb 引擎中的所有状态信息,包含了锁信息,如下:

image-20241005132944757

总共获取锁的步骤为:

1、在获取表中某行数据的独占锁之前,会先获取表的 IX 锁

2、在最大索引后边加上间隙锁,避免在 RR 隔离级别下发生幻读

3、表中只有两条记录 id = 1、id = 5,因此会在两条记录上添加 X 锁,即临键锁

Lock Mode 对应含义

在使用 show engine innodb status 查看锁时,有很多 lock mode IX 等等,列举一下锁模式对应的含义:

  • IX:代表意向排他锁
  • X:代表Next-Key Lock锁定记录本身和记录之前的间隙(X)
  • S:代表Next-Key Lock锁定记录本身和记录之前的间隙(S)
  • X, REC_NOT_GAP:代表只锁定记录本身(X)
  • S, REC_NOT_GAP:代表只锁定记录本身(S)
  • X, GAP:代表间隙锁,不锁定记录本身(X)
  • S, GAP:代表间隙锁,不锁定记录本身(S)
  • X, GAP, INSERT_INTENTION:代表插入意向锁

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

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

相关文章

WindowsTerminal 美化-壁纸随机更换

目录 一. 相关网址二. 壁纸随机更换思路三. 指定 WindowsTermina 壁纸路径四. 编写脚本,随机替换壁纸4.1 powershell脚本4.2 .bat批处理脚本 四. 配置定时任务,添加触发器五. 效果 一. 相关网址 官方下载 Windows Terminal 官方Github微软商店 美化 Oh …

链式二叉树及二叉树各种接口的实现(C)

二叉树的性质 若规定根节点的层数为1,则一棵非空二叉树的第 i i i层上最多有 2 i − 1 2^{i-1} 2i−1个结点.若规定根节点的层数为1,则深度为h的二叉树的最大结点数是 2 h − 1 2^{h}-1 2h−1对任何一棵二叉树,如果度为0其叶结点个数为 n 0 …

Semantic Communication Meets Edge Intelligence——构造终端共享的知识图谱指导无线物联网通信中文本的传输

论文链接: IEEE Xplore Full-Text PDF:https://ieeexplore.ieee.org/stamp/stamp.jsp?tp&arnumber9979702 1. 背景 随着自动驾驶、智能城市等应用的发展,移动数据流量将大幅增加。传统的香农信息论(CIT)通信系统已接近其带…

内网穿透工具ngrok

写作背景 最近在公司内购淘了个MAC电脑,想当个Linux服务器起Docker搭建环境用,现在问题是如何在公网上能访问到MAC这个机器上的资源。 之前写了一篇文章Mac当作云服务器,你真的会搞吗 最近想重启一下这台老伙计了,发现ngrok还是…

CIKM 2024 | 时空数据(Spatial-temporal)论文总结

CIKM 2024于10月21号-10月25号在美国爱达荷州博伊西举行(Boise, Idaho, USA) 本文总结了CIKM 2024有关时空数据(spatial-temporal data)的相关论文,主要包含交通预测,插补,事故预测&#xff0c…

计算机网络——http和web

无状态服务器——不维护客户端 怎么变成有状态连接 所以此时本地建立代理—— 若本地缓存了——但是服务器变了——怎么办?

CSS元素堆叠

通常我们可能会认为 HTML 网页是个二维的平面,因为页面中的文本、图像或者其它元素都是按照一定顺序排列在页面上的,每个元素之间都有一定的间隙,不会重叠。然而,实际的网页其实是三维的,元素之间可能会发生堆叠&#…

《python语言程序设计》2018版第8章19题几何Rectangle2D类(中)-同志们我要起飞了

前言 昨天的原始绘制两个矩形的代码段draw_rec2原始draw_rec2运行结果我们不是上面往右转90.我怎么往左转90不对吗??? ☺️结果利用已建立完的Rectangle2D类来实现Rectangle2D类的代码可以找上集看,今天是锻炼的一天好几个倒立体式解锁了.祝大家愉快 经过昨天晚上的努力我终…

Python画笔案例-078 绘制 颜色渐变之coloradd

1、绘制纯 颜色渐变之coloradd 通过 python 的turtle 库绘制 颜色渐变之coloradd,如下图: 2、实现代码 绘制 颜色渐变之coloradd,以下为实现代码: """颜色渐变之coloradd.py本程序需要coloradd模块支持,请在cmd窗口,即命令提示符下输入pip install colorad…

VMware桥接模式无法连接网络

windows下打开控制面板,找到WLAN,记住下面的名称(带有VMware的都是虚拟机的网卡,要找到物理主机的网卡) 回到VMware,编辑——打开虚拟网络编辑器 桥接选择上面的WLAN下的网络名称,确定即可。&…

tortorise数据库迁移变化aerich

数据库迁移 使用场景,当需要修改定义的数据库中表的数据时,就可以利用aerich进行迁移改动 例如 class Asset(models.Model):aid fields.CharField(max_length50, pkTrue)asset_name fields.CharField(max_length150)target_name fields.CharField(…

GO网络编程(三):海量用户通信系统1:登录功能初步

一、准备工作 需求分析 1)用户注册 2)用户登录 3)显示在线用户列表 4)群聊(广播) 5)点对点聊天 6)离线留言 主界面 首先,在项目根目录下初始化mod,然后按照如下结构设计目录: 海量用户通信系统/ ├── go.mod ├── client/ │ ├──…

血液细胞计数与检测(BCCD)数据集教程

BCCD 数据集:血液细胞检测与计数-CSDN博客文章浏览阅读431次,点赞5次,收藏3次。BCCD 数据集:血液细胞检测与计数 BCCD_Dataset BCCD (Blood Cell Count and Detection) Dataset is a small-scale dataset for blood cells detecti…

U盘格式化别担心,数据恢复神器来了!

一、恢复数据的紧迫性和希望 别担心,小编我有幸深陷U盘数据丢失的境地,因此通过不懈的努力与反复试验,今日在此为广大读者分享一次趟雷后恢复数据工具的真实体验;当U盘格式化后,你可能会面临数据的丢失,但…

超简单 Flux 换背景工作流:结合最新 ControlNet 深度 Depth 模型

在本篇文章中,我们将深入探讨如何使用 Flux ControlNet Depth 模型进行换背景。这种方法是我之前基于 Flux 模型换背景工作流的简化版。虽然旧的工作流程功能强大,但它非常复杂且运行缓慢。今天,我们将学习一个更快速、更易用的替代方案。 F…

变电站红外检测数据集 1180张 变电站红外 标注voc yolo 13类

变电站红外检测数据集 1180张 变电站红外 标注voc yolo 13类 变电站红外检测数据集 名称 变电站红外检测数据集 (Substation Infrared Detection Dataset) 规模 图像数量:1185张图像。类别:13种设备类型。标注个数:2813个标注。 数据划分…

【全球顶级域名后缀】

数据时间: 2024.10.6 广告: 五分钟申请SSL证书 (手机电脑都能用) ["aaa","aarp","abarth","abb","abbott","abbvie","abc","able","abogado","abudhabi","ac"…

一项研究表明,只需一滴干血,新的检测技术或许可以在几分钟内发现癌症

每周跟踪AI热点新闻动向和震撼发展 想要探索生成式人工智能的前沿进展吗?订阅我们的简报,深入解析最新的技术突破、实际应用案例和未来的趋势。与全球数同行一同,从行业内部的深度分析和实用指南中受益。不要错过这个机会,成为AI领…

视频加字幕用什么软件最快?12款工具快速添加字幕!

对于大多数同学来讲,剪辑中比较头疼的就是如何给视频加字幕和唱词啦,特别是用Pr或者FCXP等专业剪辑软件,加字幕也是特别费时的,哪怕是有批量添加的功能orz... 虽然关于这方面的内容已经很多啦,但是真正全面的内容还特…

unity 2d 近战攻击判定的三种方式以及精确获取碰撞点

精确获取碰撞点 核心是获取武器碰撞盒最顶点&#xff0c;然后获取敌人碰撞盒距离该点最近的点 /// <summary>/// 获取获取武器前端位置 碰撞盒最左或最右顶点/// </summary>/// <param name"collider"></param>/// <param name"…