MySQL锁篇

news2024/10/3 4:30:18

文章目录

      • 说明:
      • 锁篇
          • 一、MySQL有那些锁?
          • 二、MySQL 是怎么加锁的?
          • 三、update 没加索引会锁全表?
          • 四、MySQL 记录锁+间隙锁可以防止删除操作而导致的幻读吗?
          • 五、MySQL 死锁了,怎么办?
          • 六、字节面试:加了什么锁,导致死锁的?

说明:

此类文章是为小林coding的图解MySQL,所简写,目的在于大家更快抓到小林文章的重点
本文全部由我简化,但是其中有部分引用小林的文章内容
希望大家掌握精髓,构建知识体系和知识框架

锁篇

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-ZLBwp08k-1677200893814)(../my_images/1e37f6994ef44714aba03b8046b1ace2.png)]

一、MySQL有那些锁?

全局锁

全局锁主要应用于做全库逻辑备份

缺点是:无法更新数据,会造成业务停滞,不过可重复读隔离级别和MVCC解决了这个问题
备份数据库的工具是 mysqldump,在使用 mysqldump 时加上 –single-transaction 参数的时候,就会在备份数据库之前先开启事务。这种方法只适用于支持「可重复读隔离级别的事务」的存储引擎。

# 使用全局锁
flush tables with read lock # 整个数据库就处于只读状态了
# 释放
unlock tables

表级锁

表锁
元数据锁 MDL
意向锁
AUTO-INC锁

# 表锁
# 表级别的共享锁,也就是读锁;
lock tables t_student read;
# 表级别的独占锁,也就是写锁;
lock tables t_stuent write;
# 释放所有锁,当会话退出后,也会释放所有表锁。表锁的颗粒度太大,会影响并发性能,尽量避免使用,InnoDB 牛逼的地方在于实现了颗粒度更细的行级锁。
unlock tables

# 元数据锁,对数据库表进行操作时,会自动给这个表加上 MDL
# 对一张表进行 CRUD 操作时,加的是 MDL 读锁;
# 对一张表做结构变更操作的时候,加的是 MDL 写锁;
# 有读锁时候想要加写锁,是会被堵塞的,直到读锁被释放,才能加上写锁,相反也是
# MDL 是在事务提交后才会释放,这意味着事务执行期间,MDL 是一直持有的。
# 读读写读读...
# 可可堵堵堵...

# 意向锁
# 意向 共享/独占锁 + 共享/独占锁 # 也就是,当执行插入、更新、删除操作,需要先对表加上「意向独占锁」,然后对该记录加独占锁。
# 而普通的 select 是不会加行级锁的,普通的 select 语句是利用 MVCC 实现一致性读,是无锁的
# select 也是可以对记录加共享锁和独占锁的,锁定读
//先在表上加上意向共享锁,然后对读取的记录加共享锁
select ... lock in share mode;
//先表上加上意向独占锁,然后对读取的记录加独占锁
select ... for update;
# 意向锁的目的是为了快速判断表里是否有记录被加锁。

#AUTO-INC 锁
# 主键自增AUTO_INCREMENT ,主要是通过AUTO-INC 锁实现的
# 执行完插入语句后就会立即释放
# 一个事务在持有 AUTO-INC 锁的过程中,其他事务想要再插入该表,就会被阻塞,为了保证连续递增
# 对大量数据进行插入的时候,会影响插入性能,InnoDB 存储引擎提供了一种轻量级的锁来实现自增
# 给该字段赋值一个自增的值,就把这个轻量级锁释放了,而不需要等待整个插入语句执行完后才释放锁
# InnoDB 存储引擎提供了个 innodb_autoinc_lock_mode 的系统变量,是用来控制选择用 AUTO-INC 锁
# = 0,就采用 AUTO-INC 锁,语句执行结束后才释放锁;
# = 2,就采用轻量级锁,申请自增主键后就释放锁,并不需要等语句执行后才释放。
# = 1
# 	普通 insert 语句,自增锁在申请之后就马上释放
# 	类似 insert … select 这样的批量插入数据的语句,自增锁还是要等语句结束后才被释放
# 当 innodb_autoinc_lock_mode = 2 是性能最高的方式,但是当搭配 binlog 的日志格式是 statement 一起使用的时候,在「主从复制的场景」中会发生数据不一致的问题。
# 当 innodb_autoinc_lock_mode = 2 时,并且 binlog_format = row,既能提升并发性,又不会出现数据一致性问题

表锁和行锁是满足读读共享、读写互斥、写写互斥的

行级锁
MyISAM 引擎并不支持行级锁。

  • Record Lock,记录锁,也就是仅仅把一条记录锁上;
  • Gap Lock,间隙锁,锁定一个范围,但是不包含记录本身;
  • Next-Key Lock:临键锁Record Lock + Gap Lock 的组合,锁定一个范围,并且锁定记录本身。

间隙锁之间是兼容的,即两个事务可以同时持有包含共同间隙范围的间隙锁,并不存在互斥关系,因为间隙锁的目的是防止插入幻影记录而提出的

因为记录锁是会出现不兼容的,而临键锁是Record Lock + Gap Lock 的组合,自然也是会不兼容的,与记录锁一样

一个事务在插入一条记录的时候,该位置已经被其他事务加了间隙锁(包含临键锁),就会发生阻塞,在拥有间隙锁的那个事务提交前,会生成一个插入意向锁
插入意向锁名字虽然有意向锁,但是它并不是意向锁,它是一种特殊的间隙锁,属于行级别锁

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-ZmmoGp7N-1677200893815)(../my_images/image-20230223084901512.png)]

二、MySQL 是怎么加锁的?

什么 SQL 语句会加行级锁?

普通的 select 语句是不会对记录加锁的,因为它属于快照读,是通过 MVCC(多版本并发控制)实现的。除非加锁定读

update 和 delete 操作都会加行级锁,且锁的类型都是独占锁(X型锁)

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-6pWX3BwP-1677200893815)(../my_images/image-20230223085937788.png)]

怎么加行级锁的?

加锁的对象是索引,加锁的基本单位是 next-key lock,前开后闭区间
间隙锁是前开后开区间
在能使用记录锁或者间隙锁就能避免幻读现象的场景下, next-key lock 就会退化成退化成记录锁或间隙锁

select * from performance_schema.data_locks\G; 这条语句,查看事务执行 SQL 过程中加了什么锁。

  • 如果 LOCK_MODE 为 X,说明是 next-key 锁;
  • 如果 LOCK_MODE 为 X, REC_NOT_GAP,说明是记录锁;
  • 如果 LOCK_MODE 为 X, GAP,说明是间隙锁;

之后再补充五种不同的请况是如何加行级锁

三、update 没加索引会锁全表?

在 update 语句的 where 条件没有使用索引,就会全表扫描,于是就会对所有记录加上 next-key 锁(记录锁 + 间隙锁),相当于把整个表锁住了

除了 select ... from语句,其他语句都会被锁住不能执行,业务会因此停滞,接下来等着你的,就是老板的挨骂

那 update 语句的 where 带上索引就能避免全表记录加锁了吗?-并不是。
关键还得看这条语句在执行过程种,优化器最终选择的是索引扫描,还是全表扫描,如果走了全表扫描,就会对全表的记录加锁了

如何避免这种事故的发生?

我们可以将 MySQL 里的 sql_safe_updates 参数设置为 1,开启安全更新模式。
update 语句必须满足如下条件之一才能执行成功:

  • 使用 where,并且 where 条件中必须有索引列;
  • 使用 limit;
  • 同时使用 where 和 limit,此时 where 条件中可以没有索引列;
四、MySQL 记录锁+间隙锁可以防止删除操作而导致的幻读吗?

当然了,就像增删改查,都可以防止,查是select … for update ,这都是当前读

举个例子哈,执行了select … for update,就是开启了一个事务,对这个区间的记录加了next-key lock锁

然后执行删除操作,这时候,因为有锁就被阻塞了,加意向锁,进入等待状态

但是记住,一定要检查,这个字段是否加了索引,否则会走全表扫面,锁全表

五、MySQL 死锁了,怎么办?

两个事务,分别执行了select … for update,后执行更新或者删除,创建,就会进入死锁,双方都在等待对方提交事务,无限循环,前提,没开死锁检查

如何避免死锁?

两种策略通过「打破循环等待条件」
数据库层面,两种策略通过「打破循环等待条件」

  • 设置事务等待锁的超时时间,innodb_lock_wait_timeout 是用来设置超时时间的,默认值时 50 秒。
  • 开启主动死锁检测,检测在发现死锁后,主动回滚死锁链条中的某一个事务,让其他事务得以继续执行,将参数 innodb_deadlock_detect 设置为 on,默认就开启
六、字节面试:加了什么锁,导致死锁的?

两个事务,先分别执行了update,再执行insert,出现死锁

update时候是x型意向锁,x型间隙锁

insert时候是x型插入意向锁,x型间隙锁

插入意向锁和间隙锁之间是冲突的

双方都在等待对方的间隙锁释放,造成循环等待
满足了互斥、占有且等待、不可强占用、循环等待

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

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

相关文章

【单例模式】单例模式创建的几种方式

一、饿汉模式饿汉模式是在类加载的时候就初始化了一份单例对象,所以他不存在线程安全问题。优点:不存在线程安全问题,天然的线程安全缺点:在类加载的时候就已经创建了对象,如果后续代码里没有使用到单例,就…

跟20%的同行去竞争80%的蓝海市场不香吗?

近年来,由于科技的发展等诸多因素,跨境电商行业有了长足的发展空间,不少人也有想要入行的打算。对于不是很了解这一行业的新手来说,如何选择合适的跨境电商市场与平台就显得至关重要。 一直以来,作为全球第四大电商市…

Android自定义View实现横向的双水波纹进度条

效果图:网上垂直的水波纹进度条很多,但横向的很少,将垂直的水波纹改为水平的还遇到了些麻烦,现在完善后发布出来,希望遇到的人少躺点坑。思路分析整体效果可分为三个,绘制圆角背景和圆角矩形,绘…

阅读HAL源码之重点总结

HAL封装中有如下特点(自己总结的): 特定外设要设置的参数组成一个结构体; 特定外设所有寄存器组成一个结构体; 地址基本都是通过宏来定义的,定义了各外设的起始地址,也就是对应寄存器结构体的地…

问答系统(QA)调研

引言 智能问答系统广泛用于回答人们以自然语言形式提出的问题,经典应用场景包括:智能语音交互、在线客服、知识获取、情感类聊天等。根据QA任务,可以将QA大致分为5大类,分别为: 文本问答(text-based QA&am…

使用Chemistry Development Kit (CDK) 来进行化学SMILES子结构匹配

摘要 SMILES是一种用于描述化合物结构的字符串表示法,其中子结构搜索是在大规模化合物数据库中查找特定的结构。然而,这种搜索方法存在一个误解,即将化合物的子结构视为一个独立的实体进行搜索,而忽略了它们在更大的化合物中的上…

码匠 × OpenAI :快速生成 SQL 语句,提升开发效率!

目录 使用 OpenAI 生成 SQL 码匠连接与集成 OpenAI 总结 关于码匠 在码匠中,编写 SQL 语句,并结合码匠一系列开箱即用的组件实现复杂的业务逻辑,是很常见的应用开发场景。然而,不同的数据库在 SQL 增删改查操作语法、类型字段和…

【C++进阶】一些小知识点

const限定符 用const给字面常量起个名字(标识符),这个标识符就称为标识符常量;因为标识符常量的声明和使用形式很像变量,所以也称常变量。声明方式: const int a 77; const float PI 3.14159f&#xff…

单核CPU, 1G内存,也能做JVM调优吗?

最近,笔者的技术群里有人问了一个有趣的技术话题:单核CPU, 1G内存的超低配机器,怎么做JVM调优?这实际上是两个问题。单核CPU的超低配机器,怎么充分利用CPU?单核CPU, 1G内存的超低配机器,怎么做J…

python学习之OpenCV-Python模块的部分应用示例(生成素描图和动漫图)

文章目录前言一、图片转灰度二、对图片进行二值化处理三、对图片去除噪点四、调整图片透明度五、生成素描滤镜效果图(方法结合应用)六、生成动漫卡通滤镜效果图(方法结合应用)总结前言 OpenCV 是一个图像和视频处理库&#xff0c…

由Geoscene Enterprise 2.1 升级至Geoscene Enterprise 3.1

文章目录一、升级前工作二、升级Geoscene portal三、升级Web Adaptor(针对portal门户)四、升级Server 站点(作为门户托管服务器)五、升级Web Adaptor(针对server)六、升级Data Store需求:由GeoS…

bestSync外网转内网操作

一.外网笔记本操作 设置管理员密码打开笔记本电脑,设置管理员密码(如果已经设置了的,请忽略该操作):左下角鼠标左键点击->控制面板->用户帐户和家庭安全->用户帐户->密码,密码设置完成后回到桌…

【Acwing 周赛复盘】第90场周赛复盘(2023.2.11)

【Acwing 周赛复盘】第90场周赛复盘(2023.2.11) 周赛复盘 ✍️ 本周个人排名:1488/2884 AC情况:1/3 这是博主参加的第五次周赛,这次做题的时候,感觉题目好难 😂 但是一听y总讲解,又…

ORA error集锦

1、oralce 数据客户端需要安装的问题 保存信息为: “无法连接到数据库,因为数据库客户端软件无法加载。确保已正确安装并配置数据库客户端软件” 从百度网盘下载,并安装win32 oracle client 安装包 2、ORA错误 “执行异常,ORA-00911: inval…

TCP四次挥手

TCP 四次挥手过程是怎样的? TCP 断开连接是通过四次挥手方式。 双方都可以主动断开连接,断开连接后主机中的「资源」将被释放,四次挥手的过程如下图: 客户端打算关闭连接,此时会发送一个 TCP 首部 FIN 标志位被置为 1…

Revit标注问题:尺寸界线长度和“快速尺寸定位标注”

一、 Revit中关于标注的问题 1.有时候,但我们建完模型要进行一定的标注,往往会出现这样的间题,如图1所示 按照正常的标注来说,通常它的标注的正确方式应该是这样的,尺寸界线长度应该是指向图元的,如图2所示…

【源码系列】Faster RCNN源码详解(一)——transform

系列文章目录 文章目录系列文章目录前言一、transform二、总结1.标准化2.缩放3.batch前言 Faster RCNN的源码整体可以分为7个模块,每个模块负责不同的功能。推荐B站up霹雳吧啦Wz讲解的Faster RCNN源码,已经很详细了,这里只是个人的一些理解总…

【Unity VR开发】结合VRTK4.0:创建滑块

语录: 只有经历地狱般的磨练,才能炼出创造天堂的力量。 前言: 滑块是一个非常简单的控件,它允许通过沿有限的驱动轴滑动 Interactable 来选择不同的值。我们将使用线性驱动器创建一个滑块控件,该控件允许我们根据与滑…

蓝桥杯刷题五

1.01背包问题这题就是01背包问题的模板题 回顾一下01背包 01就是这个东西选和不选01背包的表达式是f[i]max(f[i-v]w,f[i]);那么这题就可以直接做了 值得注意的是这里只用了一维数组 所以更新的时候要从后往前面更新#include <bits/stdc.h> using namespace std; const in…

【JDK8新特性之Stream流-Stream结果收集案例实操】

一.JDK8新特性之Stream流-Stream结果收集以及案例实操 二.Stream结果收集(collect函数)-实例实操 2.1 结果收集到集合中 /*** Stream将结果收集到集合中以及具体的实现 collect*/Testpublic void test01(){// 收集到List中 接口List<Integer> list Stream.of(1, 2, 3…