MySQL学习[4] ——MySQL锁

news2025/1/12 12:14:33

四、MySQL锁

4.1 MySQL有哪些锁?

4.1.1 全局锁

全局锁就是**对整个数据库实例加锁,主要用于全库逻辑备份**等场景。

flush tables with read lock # 加全局锁

unlock tables   # 解锁

加上全局(读)锁后,整个数据库都是只读状态。若数据库的数据较多,导致整个处理流程较慢,数据库长时间为只读状态,造成业务停滞、服务长时间不可用。

因此,由于InnoDB支持事务,支持可重复读的隔离级别,在备份数据库之前先开启事务。整个事务执行期间都在用这个 Read View,而且由于 MVCC 的支持,备份期间业务依然可以对数据进行更新操作。而如MylSAM等不支持事务的引擎,就只能通过全局锁的方式

4.1.2 表级锁

MySQL中存在多种表级锁:

  • 表锁

    锁住整张表,命令如下:

    # 对表t_student加锁
    lock tables t_student read; 	# 表读锁
    lock tables t_student write; 	# 表读锁
    
    unlock tablse # 释放锁
    

    表锁除了会限制别的线程的读写外,也会限制本线程接下来的读写操作。

    如果本线程对学生表加了「共享表锁」,那么本线程接下来如果要对学生表执行写操作的语句,是会被阻塞的,当然其他线程对学生表进行写操作时也会被阻塞,直到锁被释放。

    尽量避免在使用 InnoDB 引擎的表使用表锁,因为表锁的颗粒度太大,会影响并发性能,InnoDB 牛逼的地方在于实现了颗粒度更细的行级锁

  • 元数据锁(meta data lock,MDL)

    无需显示地调用MDL,每次访问一张表时会自动加上。MDL 是为了保证当用户对表执行 CRUD(增删改查) 操作时,防止其他线程对这个表结构做了变更。MDL 是在事务提交后才会释放,这意味着**事务执行期间,MDL 是一直持有的**。

    因此如果存在一个长事务对某个表加上了MDL读锁,如果此时有线程尝试修改这个表的结构,但无法拿到MDL写锁,则陷入阻塞。此后任何想要读或者写的线程都无法执行而是阻塞等待(因为申请 MDL 锁的操作会形成一个队列,队列中写锁获取优先级高于读锁)。

    所以为了能安全的对表结构进行变更,在对表结构变更前,先要看看数据库中的长事务,是否有长事务已经对表加上了 MDL 读锁,如果可以考虑 kill 掉这个长事务,然后再做表结构的变更。

  • 意向锁

    意向锁是指示一个事务**在未来可能会请求对某些资源的锁定**。

    • 在使用 InnoDB 引擎的表里对某些记录加上「共享锁」之前,需要先在**表级别**加上一个「意向共享锁」;
    • 在使用 InnoDB 引擎的表里对某些纪录加上「独占锁」之前,需要先在**表级别**加上一个「意向独占锁」;

    当执行插入、更新、删除操作,需要先对表加上「意向独占锁」,然后对该记录加独占锁。查询不需要,因为查询是通过MVCC实现一致性读的,无需加锁。

    意向共享锁和意向独占锁是表级锁,不会和行级的共享锁和独占锁发生冲突,而且意向锁之间也不会发生冲突,只会和共享表锁(lock tables ... read)和独占表锁(lock tables ... write)发生冲突。

    如果没有「意向锁」,那么加「独占表锁」时,就需要遍历表里所有记录,查看是否有记录存在独占锁,这样效率会很慢。那么有了「意向锁」,由于在对记录加独占锁前,先会加上表级别的意向独占锁,那么在加「独占表锁」时,直接查该表是否有意向独占锁,如果有就意味着表里已经有记录被加了独占锁,这样就不用去遍历表里的记录。意向锁的目的是为了快速判断表里是否有记录被加锁

  • AUTO-INC锁(自增锁)

    一个表中的主键通常是自增的,在插入数据时,数据库会自动给主键赋值递增的值,这主要是通过AUTO-INC锁来实现的在插入数据时,会加一个表级别的 AUTO-INC 锁,然后为被 AUTO_INCREMENT 修饰的字段赋值递增的值,等插入语句执行完成后,才会把 AUTO-INC 锁释放掉。

    AUTO-INC 锁是特殊的表锁机制,锁不是再一个事务提交后才释放,而是再执行完插入语句后就会立即释放

4.1.3 行级锁

InnoDB 引擎是支持行级锁的,而 MyISAM 引擎并不支持行级锁。行级锁的类型主要有三类:

  • 记录锁(Record Lock)

    将一条记录加锁,又分为共享锁(读锁、S锁)排他锁(读锁、X锁),不同事物之间可能存在冲突关系。

  • 间隙锁(Gap Lock)

    锁定一个范围,不包含记录,间隙锁不存在互斥关系(因为不涉及到具体记录,于是不同事务可以同时包含共同范围的间隙锁),只存在于可重复读隔离级别下,用来防止可重复读隔离级别下的幻读现象。

  • 临键锁(Next-Key Lock)

    就是Record Lock和Gap Lock的组合,锁定记录本身和一个范围。即能**保护该记录,又能阻止其他事务将新纪录插入到被保护记录前面的间隙中**。因此,虽然间隙锁是多个事务相互兼容的,但记录锁会存在冲突关系。

还有一种特殊的间隙锁插入意向锁

它是指当一个插入操作发现插入的位置被加了间隙锁,那么这个线程想要向这个区域加上一个“插入意向锁”,只能阻塞等待,直到间隙锁被释放。

4.1.4 乐观锁与悲观锁
  • 乐观锁

    一种思想,认为对同一个数据的并发操作发生概率较小,不需要每次都对数据上锁。常通过时间戳、版本号机制来实现。适用于读操作比较多的场景。

  • 悲观锁

    一种思想,认为总是会发生并发冲突,具有强烈独占性和排他性,通过锁机制来保护数据。适用于写操作比较多的场景。

4.2 MySQL是如何加行级锁的?

4.2.1 什么SQL语句会加行级锁?

InnoDB 引擎支持行级锁,与表级锁相比,行级锁的并发性能要好很多。

普通的 select 语句是不会对记录加锁的(除了串行化隔离级别),因为它属于快照读,是通过 **MVCC(多版本并发控制)实现**的。但是可以通过显式指定的方式给select语句加行级锁:

select ... lock in shared mode;    # 对读取的记录加共享锁
select ... for update;			   # 对读取的记录加排他锁

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

4.2.2 InnoDB两阶段锁协议

在InnoDB引擎中,可重复读隔离级别下,行级锁遵顼两阶段锁协议:在需要的时候加上,事务提交或出现回滚时才会释放(而不是操作完成了立即释放)。

4.2.2 MySQL行级锁加锁规则

MySQL中(InnoDB引擎)行级锁**加锁的对象是索引,加锁的基本单位是Next-Key Lock**。在一些场景下,Next-Key Lock会退化成记录锁或间隙锁。

在能使用记录锁或者间隙锁就能避免幻读现象的场景下, next-key lock 就会退化成记录锁或间隙锁

  • 唯一索引等值查询

    若记录「存在」,在索引树上定位到这一条记录后,将该记录的索引中的 next-key lock 会退化成「记录锁」

    若记录「不存在」,在索引树找到第一条大于该查询记录的记录后,将该记录的索引中的 next-key lock 会退化成「间隙锁」

  • 唯一索引范围查询

    当唯一索引进行范围查询时,会对每一个扫描到的索引加 next-key 锁,在一些情况下的索引的next-key锁会退化为记录锁或间隙锁。

  • 非唯一索引等值查询

    当我们用非唯一索引进行等值查询的时候,因为存在两个索引,一个是主键索引,一个是非唯一索引(二级索引),所以在加锁时,同时会对这两个索引都加锁,但是对主键索引加锁的时候,只有满足查询条件的记录才会对它们的主键索引加锁。针对非唯一索引等值查询时,对于扫描到的二级索引记录加 next-key 锁,在某些情况下的二级索引的锁会退化为间隙锁。

  • 非唯一索引范围查询

    非唯一索引和主键索引的范围查询的加锁也有所不同,不同之处在于非唯一索引范围查询,索引的 next-key lock 不会有退化为间隙锁和记录锁的情况,也就是非唯一索引进行范围查询时,对二级索引记录加锁都是加 next-key 锁

4.2.3 没有索引的查询会发生什么?

对于存在索引的查询,查询语句都有使用索引查询,也就是查询记录的时候,是通过索引扫描的方式查询的,然后对扫描出来的记录进行加锁

如果**锁定读查询语句,没有使用索引列作为查询条件,或者查询语句没有走索引查询,导致扫描是全表扫描。那么,每一条记录的索引上都会加 next-key 锁,这样就相当于锁住的全表,这时如果其他事务对该表进行增、删、改操作的时候,都会被阻塞**。

注意:这里是说对整张表的索引都加锁,而不是对表加锁。

4.2.4 可重复读隔离级别中Next-Key Lock可以防止删除操作导致的幻读吗?

前面说到,在可重复读隔离级别中,通过使用「记录锁+间隙锁」可以很大概率上避免新插入数据带来的幻读现象。

这种方案可以防止删除操作带来的幻读现象吗?

可以大概率避免,因为**当前读**的语句会对索引加Next-Key Lock,其他事务对被加锁的记录和间隙上增、删、改的操作都会被阻塞。

4.2.5 MySQL中加了什么锁会导致死锁?

死锁的四个条件:互斥、占有且等待、不可强占用、循环等待

由于间隙锁不会互斥,而当想要插入数据时,如果某个范围正好存在间隙锁,那么这个插入事务会向这个区域加上一个**“插入意向锁”并等待,直到间隙锁被释放**。

如上图所示,事务A和事务B在分别执行完update语句后,都含有一个间隙锁(事务结束后才会释放锁)。从下图的表中数据可以分析得到,两个事务的间隙锁的范围都是(20,30)

外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传

因此,两个事务分别向对方持有的间隙锁范围内插入一条记录,而**插入操作为了获取到插入意向锁,都在等待对方事务的间隙锁释放**,于是就造成了循环等待,满足了死锁的四个条件,因此发生了死锁。

4.2.6 如何避免死锁?
  • 破坏死锁条件

    互斥、占有且等待、不可强占用、循环等待。只要系统发生死锁,这些条件必然成立,但是只要破坏任意一个条件就死锁就不会成立。

  • 设置事务等待锁超时时间

    当一个事务阻塞等待时间超过阈值后,直接对该事务进行回滚,避免死锁。

  • 开启死锁检测

    MySQL支持死锁检测,开启后当发现死锁时,会主动回滚死锁链条中的某个事务,解除死锁。

4.3 InnoDB使用表锁还是行锁?

为了保证并发性能,InnoDB引擎在绝大部分情况使用行级锁

使用表级锁的情况:

  • 表比较大,需要对表中全部或大部分数据进行更新;
  • 事务涉及到多个表,比较复杂,行锁可能会引起死锁,导致事务大量回滚。

资料参考

内容大多参考自:图解MySQL介绍 | 小林coding (xiaolincoding.com)

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

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

相关文章

网络安全-第二阶段-linux操作系统01

一. linux介绍: windows,mac,linux都是由unix系统发展而来。 linux:类unix系统; 二. Centos系统的安装: 可以去清华大学开源软件镜像站下载: 输入ip addr: 可以看到自己电脑的ip地址; 1. ssh远程连接linux: 使用windows或者linux的物理机或者虚拟机都可以连接上它,…

MPU6050+OLED读取姿态角(超级细讲)

STM32F103C8T6读取陀螺仪MPU6050的角度数据,使用6050自带DMP库姿态解算出各个方向的角度,并使用OLED实时刷新显示,同时可以将数据通过串口发送到计算机,每一组数据50ms。本操作过程简单,方便移植,显示屏接P…

ppt中添加页码(幻灯片编号)及问题解决方案

在幻灯片母版中,选择插入 幻灯片编号 右下角显示幻灯片编号 问题一:母版中没有显示编号 原因可能是母版版式中没有设置显示,勾选即可。 问题二:子母版中没有显示幻灯片 将母版中的编号复制到子母版中。 问题三:应用…

Element-UI自学实践

概述 Element-UI 是由饿了么前端团队推出的一款基于 Vue.js 2.0 的桌面端 UI 组件库。它为开发者提供了一套完整、易用、美观的组件解决方案,极大地提升了前端开发的效率和质量。本文为自学实践记录,详细内容见 📚 ElementUI官网 1. 基础组…

2024年7月文章一览

2024年7月编程人总共更新了5篇文章: 1.2024年6月文章一览 2.《Programming from the Ground Up》阅读笔记:p19-p48 3.《Programming from the Ground Up》阅读笔记:p49-p74 4.《Programming from the Ground Up》阅读笔记:p75…

深入理解Kafka核心设计与实践原理_03

深入理解Kafka核心设计与实践原理_03 03_消费者3.1消费者与消费者组3.2客户端开发3.2.1 必要的参数配置3.2.2 订阅主题与分区 草稿 03_消费者 与生产者对应的是消费者,应用程序可以通过KafkaConsumer来订阅主题,并从订阅的主题中拉取消息。不过在使用Ka…

Redis17-服务端优化

目录 持久化配置 慢查询 什么是慢查询 如何查看慢查询 命令及安全配置 内存配置 集群优化 持久化配置 Redis的持久化虽然可以保证数据安全,但也会带来很多额外的开销,因此持久化请遵循下列建议: 用来做缓存的Redis实例尽量不要开启持…

springboot项目打包jar 并打包为exe启动

springboot项目打包jar 并打包为exe启动(在无jdk环境下运行) 环境 SpringBoot Windows IDEA 实现 1.springboot打包为可执行jar(这里使用maven install) maven工具栏选择项目->Plugins ->install 注:如果…

Golang | Leetcode Golang题解之第332题重新安排行程

题目: 题解: func findItinerary(tickets [][]string) []string {var (m map[string][]string{}res []string)for _, ticket : range tickets {src, dst : ticket[0], ticket[1]m[src] append(m[src], dst)}for key : range m {sort.Strings(m[key])…

python对接vertx中踩的坑

需求 因为我们的硬件sdk只提供了python的版本,故需要python作为采集端来获取数据,然后将数据发送给java作为数据中心处理。 分析 这里就涉及到跨语言跨进程的数据的中转。有以下的几种解决方法 tcp:基于tcp自己拆包粘包,做心跳。一看就pa…

PV 与 PVC 状态迁移

文章目录 一、概述1、PV2、PVC 二、状态变化三、实例1、单独创建 PV1.1、创建并应用 PV1.2、查看刚创建的 PV 状态 2、单独创建 PVC2.1、创建并应用 PV2.2、查看刚创建的 PVC 状态 3、等待绑定4、删除 PV4.1、查看 PV,PVC 状态4.2、真正删除 PV4.3、查看PV PVC 状态…

PaLM-E: An Embodied Multimodal Language Model

发表时间:arXiv 6 Mar 2023 作者单位:Robotics at Google Motivation:大型语言模型已被证明可以执行复杂的任务。然而,在现实世界中启用一般推理,例如对于机器人问题,提出了落地的挑战。 解决方法&#…

探索全新AI编码代理框架:Agent Zero

引言 在科技的不断进步中,人工智能(AI)正越来越多地融入我们的日常生活。今天,我要为大家介绍一款全新的AI编码代理框架——Agent Zero。这款框架不仅可以自动化处理编码任务,还能操作文本、应用程序前端等&#xff0…

【微信小程序】页面导航和导航传参

什么是页面导航 小程序中实现页面导航的两种方式 1.声明式导航 (1). 导航到 tabBar 页面 (2). 导航到非 tabBar 页面 (3).后退导航 2.编程式导航 (1).导航到 tabBar 页面 示例代码如下: (2).导航到非 tabBar 页面 示例代码如下: (3).后退导航 示例…

WSL2Linux 子系统(十)

WSL 中启用图形界面 上一篇文章 《WSL2Linux 子系统(九)》 讲解详细讲 WSL(Windows Subsystem for Linux)解几种常见硬盘挂载方法。本篇讲述 WSL 显示画面的几种方法。WSL 本身不直接支持图形界面,在 WSL 环境中播放视频画面需要额外处理或者…

进程的执行和进程的结束

exec函数族: 作用:可以在一个程序中启动另外一个功能,用新的进程段替换当前进程的段. exec函数族用于替换当前进程的映像,允许程序在运行时加载并执行新的程序代码。 execl: l:list path:要执行的文件的路径 arg:要执行的文件的参数&a…

C#利用ffmpeg和opencv进行视频的解码播放

目录 说明 效果 项目 代码 下载 说明 利用周杰大佬的开源项目 Sdcb.FFmpeg 项目地址:https://github.com/sdcb/Sdcb.FFmpeg/ 效果 C#利用ffmpeg和opencv进行视频的解码播放 项目 代码 using OpenCvSharp; using Sdcb.FFmpeg.Codecs; using Sdcb.FFmpeg.F…

【SpringBoot】【autopoi】java生成word,基于模版生成(文本、图片、表格)

基于模版生成word 1、引入maven2、word模版编写3、java代码4、效果 AutoPoi的主要特点 参考文献 https://help.jeecg.com/autopoi/autopoi/prequel/test.html 1.设计精巧,使用简单 2.接口丰富,扩展简单 3.默认值多,write less do more 4.spring mvc支持,web导出可以简单明了 1、…

【大模型从入门到精通17】openAI API 构建和评估大型语言模型(LLM)应用5

这里写目录标题 理论问题:实践问题:理论实践 理论问题: 1.描述评估LLM应用程序输出的重要性,并提及至少三个维度,这些输出应该在这几个维度上被评估。 2.解释在评估LLM应用程序时开发稳健的性能指标的作用&#xff0c…

uniapp多图上传uni.chooseImage上传照片uni.uploadFile,默认上传9张图

uniapp多图上传uni.chooseImage上传照片uni.uploadFile 代码示例: /**上传照片 多图*/getImage() {uni.chooseImage({count: 9, //默认9sizeType: [original, compressed], //可以指定是原图还是压缩图,默认二者都有sourceType: [album], //从相册选择/…