mysql从零开始(05)----锁

news2024/11/23 21:43:55

全局锁

使用

# 启用全局锁
flush tables with read lock
# 释放全局锁
unlock tables

开启全局锁后,整个数据库就处于只读状态了,这种状态下,对数据的增删改操作、对表结构的更改操作都会被阻塞。
另外,当会话断开,全局锁也会被自动释放

应用场景

全局锁主要应用于做全库逻辑备份,这样在备份数据库期间,不会因为数据或表结构的更新,而出现备份文件的数据与预期的不一样。
如果不使用全局锁直接备份的话,可能会出现下面的情况:

用户A用户B
开始备份用户A的银行余额
向B转账
结束备份开始备份用户B的余额
这时,用户A的余额既没有减少,用户B的余额也增加了

缺点

如果数据库里有很多数据,备份就会花费很多的时间,关键是备份期间,业务只能读数据,而不能更新数据,这样会造成业务停滞。

改进

要求数据库的引擎支持的事务支持可重复读的隔离级别(隔离级别相关的知识见:隔离),如InnoDB,但是比如MyISAM不支持事务,就不能应用以下的方法。
在备份数据库之前先开启事务,会先创建Read View,然后整个事务期间都用这个Read View,而且由于MVCC的支持,备份期间业务也可以对数据库进行更新操作。
因为在可重复读的隔离级别下,即使其他事务更新了表的数据,也不会影响备份数据库时的 Read View,这就是事务的隔离性,这样备份期间备份的数据一直是在开启事务时的数据。
备份数据库的工具是 mysqldump,在使用 mysqldump 时加上 –single-transaction 参数的时候,就会在备份数据库之前先开启事务。

表级锁

MySQL 里面表级别的锁有这几种:

  • 表锁
  • 元数据锁(MDL)
  • 意向锁
  • AUTO-INC 锁

表锁

使用

对表test使用表锁

# 表级别的共享表锁,也就是读锁;
lock tables test read;

# 表级别的独占表锁,也就是写锁;
lock tables test write;

# 释放表锁
unlock tables

如果本线程对某表加了「共享表锁」,那么所有线程接下来如果要对该表执行写操作的语句,都会被阻塞,直到锁被释放。当会话退出后,也会自动释放所有表锁。
另外,尽量避免在使用 InnoDB 引擎的表使用表锁,因为表锁的颗粒度太大,会影响并发性能。

元数据锁MDL

我们不需要显式的使用 MDL,因为当我们对数据库表进行操作时,会自动给这个表加上 MDL:

  • 对一张表进行 CRUD 操作时,加的是 MDL 读锁;
  • 对一张表做结构变更操作的时候,加的是 MDL 写锁;

MDL 是为了保证当用户对表执行 CRUD 操作时,防止其他线程对这个表结构做了变更:当有线程在执行 select 语句( 加 MDL 读锁)的期间,如果有其他线程要更改该表的结构( 申请 MDL 写锁),那么将会被阻塞,直到执行完 select 语句( 释放 MDL 读锁)。反之,当有线程对表结构进行变更( 加 MDL 写锁)的期间,如果有其他线程执行了 CRUD 操作( 申请 MDL 读锁),那么就会被阻塞,直到表结构变更完成( 释放 MDL 写锁)。

MDL释放时机

MDL 是在事务提交后才会释放,事务执行期间,MDL 是一直持有的。

可能导致的问题

考虑以下场景:

线程A线程B线程C
开启事务
select(此时自动加上MDL读锁)
select(不会阻塞,因为读读事务不冲突)
修改表字段(阻塞)

在线程C阻塞后,后续对该表的所有select语句都会被阻塞,可能导致数据库连接线程用光

原因

出现上面那种情况的原因是因为MDL锁的操作会形成一个队列,因为写锁的优先级高于读锁,一旦队列中出现写锁等待,后面加入的读锁请求都将在写锁后面等着
所以,如果要对表结构进行变更,应当先查看数据库中是否有比较长的事务,如果有的话可以考虑kill掉长事务,再做表结构的变更

意向锁

在使用 InnoDB 引擎的表里对某些记录加上「共享锁」之前,需要先在表级别加上一个「意向共享锁」;在使用 InnoDB 引擎的表里对某些纪录加上「独占锁」之前,需要先在表级别加上一个「意向独占锁」
也就是,当执行插入、更新、删除操作,需要先对表加上「意向独占锁」,然后对该记录加独占锁。而普通的 select 是不会加行级锁的,普通的 select 语句是利用 MVCC 实现一致性读,是无锁的。但是,普通select语句也可以加共享锁和独占锁,如下:

//先在表上加上意向共享锁,然后对读取的记录加共享锁
select ... lock in share mode;

//先表上加上意向独占锁,然后对读取的记录加独占锁
select ... for update;

表中手动加的独占锁和共享锁
意向共享锁和意向独占锁是表级锁,不会和行级的共享锁和独占锁发生冲突(允许多个事务同时更改不同行的数据),而且意向锁之间也不会发生冲突(也允许边读边写),只会和共享表锁(lock tables … read)和独占表锁(lock tables … write)发生冲突。

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

所以,意向锁的目的是为了快速判断表里是否有记录被加锁。

AUTO_INC锁

概念

在插入数据时,可以不指定主键的值,使用AUTO_INCREMENT修饰,让mysql来自动维护主键自增,这个自增主要通过AUTO_INC锁来实现的。
在插入数据时,会自动添加一个表级别的AUTO_INC锁,然后为被AUTO_INCREMENT修饰的字段赋值,等插入语句完成后,才会把锁释放掉。既当一条事务在插入时,其他事务如果也想插入,都会被阻塞,从而保证该字段是连续递增的。相应的,这个操作也会影响插入的性能。

InnoDB

InnoDB 存储引擎提供了一种轻量级的锁来实现自增。

当它在插入数据的时候,同样会为被 AUTO_INCREMENT 修饰的字段加上轻量级锁,但是在它给该字段赋完一个自增的值之后,就把这个轻量级锁释放了,而不需要等待整个插入语句执行完后才释放锁。

行级锁

锁定读

InnoDB 引擎是支持行级锁的,而 MyISAM 引擎并不支持行级锁。这也是InnoDB的最大的优势

普通的 select 语句是不会对记录加锁的,因为它属于快照读。如果要在查询时对记录加行锁,可以使用下面这两个方式,这种查询会加锁的语句称为锁定读

# 对读取的记录加共享锁
select ... lock in share mode;

# 对读取的记录加独占锁
select ... for update;

上面这两条语句必须在一个事务中,因为当事务提交了,锁就会被释放,所以在使用这两条语句的时候,要加上 begin、start transaction 或者 set autocommit = 0。
共享锁又称S锁,满足读读共享、读写互斥,独占锁又称X锁,满足写写互斥、读写互斥

行级锁的类型主要有:

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

Record Lock

记录锁锁住的是一条记录。记录锁有S锁和X锁之分,并且满足读读共享、读写互斥、写写互斥

mysql > begin;
# X锁,之后其他事务都无法修改该记录
mysql > select * from test where id = 1 for update;

# 释放锁
mysql > commit;

Gap Lock

Gap Lock 只存在于可重复读隔离级别,目的是为了解决可重复读隔离级别下幻读的现象。
间隙锁锁住的是一个字段之间的间隙,其他事务无法向这个间隙中插入字段,如表中有一个范围 id 为(3,5)间隙锁,那么其他事务就无法插入 id = 4 这条记录了,以此防止幻读。
间隙锁虽然存在 X 型间隙锁和 S 型间隙锁,但是并没有什么区别,间隙锁之间是兼容的,即两个事务可以同时持有包含共同间隙范围的间隙锁,并不存在互斥关系

插入意向锁

一个事务在插入一条记录的时候,需要判断插入位置是否已被其他事务加了间隙锁(next-key lock 也包含间隙锁)。

如果有的话,插入操作就会发生阻塞,直到拥有间隙锁的那个事务提交为止(释放间隙锁的时刻),在此期间会生成一个插入意向锁,表明有事务想在某个区间插入新记录,但是现在处于等待状态(等待状态并不意味着事务成功获取到了锁,正常状态才是

插入意向锁并不是意向锁,它是一种特殊的间隙锁,属于行级别锁
两个事务不可能同时一个拥有间隙锁,另一个拥有该间隙的插入意向锁

Next-Key Lock

锁定一个范围,并且锁定记录本身。Next-key lock既能保护该记录,又能阻止其他事务将新纪录插到被保护记录前面的间隙中。
如表中有一个范围 id 为(3,5] 的 next-key lock,那么其他事务即不能插入 id = 4 记录,也不能修改 id = 5 这条记录。

因为Next-key lock包含记录锁,所以不能有两个事务同时获取相同范围的Next-key lock。

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

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

相关文章

【1015. 可被 K 整除的最小整数】

来源:力扣(LeetCode) 描述: 给定正整数 k ,你需要找出可以被 k 整除的、仅包含数字 1 的最 小 正整数 n 的长度。 返回 n 的长度。如果不存在这样的 n ,就返回 -1。 注意: n 不符合 64 位带…

手把手教你在winform中将文本或文件路径拖到控件中

文章目录 前言博主履历介绍:一、将txt文件的所有内容复制到 RichTextBox中二、将txt文件的一行内容移动到RichTextBox中三、将多个文件的全路径复制到 RichTextBox中四 、源码1、[Winform从入门到精通(1)——(如何年入30万&#x…

「MIAOYUN」:降本增效,赋能传统企业数字化云原生转型 | 36kr 项目精选

作为新经济综合服务平台第一品牌,36氪自2019年落地四川站以来,不断通过新锐、深度的商业报道,陪跑、支持四川的新经济产业。通过挖掘本土优质项目,36氪四川帮助企业链接更多资源,助力企业成长,促进行业发展…

分布式系统概念和设计——命名服务设计和落地经验

分布式系统概念和设计 通过命名服务,客户进程可以根据名字获取资源或对象的地址等属性。 被命名的实体可以是多种类型,并且可由不同的服务管理。 命名服务 命名是一个分布式系统中的非常基础的问题,名字在分布式系统中代表了广泛的资源&#…

C语言:指针求解鸡兔同笼问题

题目:鸡兔同笼问题 要求:使用自定义函数void calc(int h, int f,int *c,int *r) 求解鸡兔同笼问题。 h 表示总的头数,f 表示总的脚数。 例子: 输入: 5 16 输出: 2 3 分析: 在该代码中&a…

05-Docker安装Mysql、Redis、Tomcat

Docker 安装 Mysql 以安装 Mysql 5.7为例: docker pull mysql:5.7Mysql 单机 Mysql 5.7安装 启动 Mysql 容器,并配置容器卷映射: docker run -d -p 3306:3306 \--privilegedtrue \-v /app/mysql/log:/var/log/mysql \-v /app/mysql/data:…

ASP.NET Core MVC 从入门到精通之文件上传

随着技术的发展,ASP.NET Core MVC也推出了好长时间,经过不断的版本更新迭代,已经越来越完善,本系列文章主要讲解ASP.NET Core MVC开发B/S系统过程中所涉及到的相关内容,适用于初学者,在校毕业生&#xff0c…

VMware NSX-T Data Center 3.2.2.1 - 数据中心网络全栈虚拟化

请访问原文链接:https://sysin.org/blog/vmware-nsx-t-3/,查看最新版。原创作品,转载请保留出处。 作者主页:sysin.org VMware NSX-T Data Center 3.2.2.1 | 30 MAR 2023 | Build 21487560 VMware NSX-T Data Center 3.2.2 | 08 …

NOA上车「清一色」自主品牌,哪些供应商正在突围前线

随着入门级L2进入普及周期,以NOA(高速、城区)为代表的L2/L2赛道,正在成为主机厂、硬件供应商、算法及软件方案商的下一波市场制高点的争夺阵地。 高工智能汽车研究院监测数据显示,2023年1-3月中国市场(不含…

MySQL基础(十六)变量、流程控制与游标

1. 变量 在MySQL数据库的存储过程和函数中,可以使用变量来存储查询或计算的中间结果数据,或者输出最终的结果数据。 在 MySQL 数据库中,变量分为系统变量以及用户自定义变量。 1.1 系统变量 1.1.1 系统变量分类 变量由系统定义&#xff…

【Nacos在derby模式下密码忘记】使用derby的ij工具重置密码/修改密码

【问题描述】 nacos部署未用mysql,直接运行,使用了默认的derby数据库,这时候不一小心修改的密码给忘记了,无法登录 当时是部署在centos上的一个演示环境,没有采用mysql数据库,如果生产上,建议使用mysql。 …

php用户分享信息技术交流大学生论坛系统vue

系统应实现的目标 1. 提供安全、友好的操作环境:避免一些网上的不良言论,创造一个和谐的网络环境。 2. 提供发表帖子功能:注册的用户可以自由发帖,发表符合法律法规的言论。 3. 提供回复帖子功能:户可以自由回复&am…

camunda执行监听器如何使用

在Camunda工作流引擎中,执行监听器是一种机制,用于在业务流程执行期间捕获特定事件并执行相应的操作。它们可以帮助您实现一些重要的任务,例如: 1、记录或更新业务数据:当流程中的任务或事件发生时,您可以…

工程监测无线中继采集发送仪 指示灯功能说明及接口定义

工程监测NLM5无线中继采集发送仪 指示灯功能说明及接口定义 指示灯功能说明 标识 名称 状态 描述说明 备注说明 CHG 正在充电 常亮 正在充电 DON 充电完成 常亮 已充满 POW 电源指示 常亮 外部电源已连接 仅用于指示是否连接了外部电源 熄灭 无外部电源 SIG 空 RUN 运行状态 闪…

“数字社区”诞生 “智慧大脑”助力提升社区综合治理效能

近日,民政局等9部门印发的《关于深入推进智慧社区建设的意见》提出,到2025年,基本构建起网格化管理、精细化服务、信息化支撑、开放共享的智慧社区服务平台,初步打造成智慧共享、和睦共治的新型数字社区。 数字社区建设是智慧城市…

最优化理论-线性规划解的几何特征

目录 一、引言 二、线性规划的定义 三、线性规划的几何特征 1.可行域 2.最优解 3.等价约束 4.对偶问题 四、线性规划的应用 五、结论 一、引言 最优化理论是数学中的一个重要分支,它研究如何在给定的约束条件下,寻找一个最优解。其中&#xff…

【MySql】数据库索引

数据库索引 索引索引的创建索引的查看索引的删除 聚簇索引 & 非聚簇索引聚簇索引非聚簇索引 索引创建原则 索引 可以简单理解为一本书的目录信息,是为了提升查找效率而建立的 索引的创建 1、在创建一个主键、唯一键、外键时候,数据库会自动地针对查…

Express框架的安装和使用

1.Express框架简介 Node.js的web框架发展至今,第一个知名的框架为Connect框架.它类似一个中间件的脚手架.只提供逻辑,不实现具体的处理逻辑.中间件概念的引入Express框架奠定了基础. 2.Express框架的安装 安装分为局部安装和全局安装. 2.1局部安装 1.在D盘创建expressStud…

html实现开心消消乐小游戏

文章目录 1.设计来源1.1 游戏界面 2.效果和源码2.1 动态效果2.2 源代码 源码下载 作者:xcLeigh 文章地址:https://blog.csdn.net/weixin_43151418/article/details/130594511 html实现开心消消乐小游戏源码 《开心消消乐》 是一款三消游戏,游…

站群服务器和普通服务器区别

更有利于提升 站群服务器指的是对于站群系统提升客户开发设计的网络服务器,客户租服务器来置放好几个网站,许多客户以便免费在线上扩大曝出会挑选提升好几个网站。非站群服务器,基础只有置放两三个网站,并且在管理方法时也…