MySQL的表级锁,行级锁,排它锁和共享锁


Posted in MySQL onJuly 15, 2022

前言

如果我们和面试官聊到事务的问题,怎么回答呢?

先说下事务是什么,因为我们业务是比较复杂的,不可能一个sql就能解决的,涉及多个sql就组成一个事务。事务就是一组sql共同执行,要么完全成功,要么完全失败,不能出现部分成功或者部分失败的情况。一个事务有ACID特性:

  • 原子性:要么全部成功,要么全部失败,这样才能保证事务的一致性;
  • 一致性:比如银行的转账,扣除一个人的钱肯定要给另一个人加钱,不能光扣除不加,这样业务就存在问题,数据的一致性就破坏了;
  • 持久性:当我们数据commit以后,数据是先写到缓存当中,缓存中的数据还是要慢慢花时间往磁盘上写,如果此时停电了、宕机或者重启了,我们有redo log重做日志来保证数据库的持久性;
  • 隔离性:这块可以说下事务为什么要有隔离性,因为事务要允许并发执行,一个业务涉及了很多事务,而我们后台往往有很多业务,要能够让他们并发执行,如果所有的事务都是串行执行的话,那这样我们写多线程程序只有一个线程来做事情,这样效率很低。所以事务要并发执行,但是并发执行涉及了一些问题:事务的安全性&一致性并发的效率问题,我们以这两个东西为参考点,才得到了MySQL不同等级的并发/隔离,如果事务并发执行时我们完全不隔离的话,就可能会出现脏读(事务B读到了事务A还未提交的数据然后,然后用事务A未提交的数据去做计算,得到了很多其他的结果,然后事务A又把那个数据rollback掉,那么事务B计算出来的都是有问题的数据,脏读一定会出现问题)、不可重复读(以同样的条件去一个数据,然后再次去查询的时候发现数据的值有所改变,当然不可重复读也不一定会有问题,有些业务场景下是允许的,这和业务上数据的安全性和一致性是否严格有关)和幻读(在事务中按照同样的条件前后两次查询的结果数据量不同)这些问题。

那么我们为了解决事务并发执行遇到的问题就给出了事务的隔离级别:

  • 串行化,串行化完全用锁来实现,通过锁给所有事务排序,按顺序执行,这样做数据的安全性高但并发的效率很低,一般我们不会这样做的。
  • 未提交读,对于我们写的多线程程序来说,对于临界区代码段没有做任何的并发控制,虽然并发性高但数据安全性很低,未提交读还允许脏读的存在,这是有问题的所以绝对不会使用未提交读。串行化和未提交读在实际项目中是不会用到的,一般数据库引擎默认工作在已提交读和可重复读,这两个隔离级别就结合了数据的安全性&一致性和数据的并发效率,这两个是由MVCC多版本并发控制机制实现的
  • 已提交读,oracle默认工作级别。不允许读取未commit的数据,这个级别仍然允许不可重复读和虚读产生。
  • 可重复读,MySQL默认工作级别。保证事务再次读取是依然得到相同的数据,部分解决了虚读,但虚读是仍然会出现的

MySQL的表级锁,行级锁,排它锁和共享锁

MySQL的表级锁,行级锁,排它锁和共享锁

注意:

  • 事务隔离级别越高,为避免冲突所花费的性能也就越多,即效率低。
  • 在“可重复读”级别,实际上可以解决部分的虚读问题,但是不能防止update更新产生的虚读问题,要禁止虚读产生,还是需要设置串行化隔离级别。

事务隔离级别的实现原理:锁+MVCC。串行化底层实现原理是锁,锁有共享锁、排它锁、意向共享锁、意向排它锁、间隙锁和死锁,InnoDB的已提交读和可重复读的底层实现原理:MVCC(多版本并发控制),MVCC提供了一种并发读取方式,包括快照读(同一份数据会有多个版本)、当前读、undo log和redo log。MVCC是已提交读和可重复读的原理,锁是串行化的原理

MySQL的表级锁,行级锁,排它锁和共享锁

ACD特性用事务日志实现,I 特性用共享锁、排它锁、MVCC 实现。事务日志分为undo log(回滚日志) 和 redo log(重做日志)

一、表级锁&行级锁

  • 表级锁:对整张表加锁。开销小(因为不用去找表的某一行的记录进行加锁,要修改这张表,直接申请加这张表的锁),加锁快,不会出
  • 现死锁;锁粒度大,发生锁冲突的概率高,并发度低
  • 行级锁:对某行记录加锁。开销大(需要找到表中相应的记录,有搜表搜索引的过程),加锁慢,会出现死锁;锁定粒度最小,发生锁冲突的概率最低,并发度高

MySQL的表级锁,行级锁,排它锁和共享锁

MyISAM存储引擎只支持表级锁,InnoDB支持事务处理,支持行级锁,并发能力更好

MySQL的表级锁,行级锁,排它锁和共享锁

二、排它锁&共享锁

  • 排它锁:又称为X锁,写锁
  • 共享锁:又称为S锁,读锁

读读(SS)之间是可以兼容的,但是读写(SX、SX)之间,写写(XX)之间是互斥的

MySQL的表级锁,行级锁,排它锁和共享锁

1. 测试不同事务之间排它锁和共享锁的兼容性

我们先查看表SQL及内容

MySQL的表级锁,行级锁,排它锁和共享锁

查看隔离级别:

MySQL的表级锁,行级锁,排它锁和共享锁

首先开启一个事务A,给id=7的数据加上排它锁:

MySQL的表级锁,行级锁,排它锁和共享锁

在另一个客户端开启事务B:

MySQL的表级锁,行级锁,排它锁和共享锁

MySQL的表级锁,行级锁,排它锁和共享锁

给id=7不管加排它锁和共享锁都阻塞了并没有查询出来,因为A事务给id=7这一行的数据加了排它锁,就是写锁,其他人不能读也不能写。

总结:不同事务之间对于数据的锁,只有SS锁可以共存,XX、SX、XS都不能共存

2. 测试行锁加在索引项上

其实行锁是加在索引树上的。

MySQL的表级锁,行级锁,排它锁和共享锁

每次做完测试都把刚做的rollback。

用表的无索引字段作为过滤条件

MySQL的表级锁,行级锁,排它锁和共享锁

那现在事务2获取不同行chenwei的记录

MySQL的表级锁,行级锁,排它锁和共享锁

InnoDB是支持行锁的,刚才以主键id为过滤条件时,事务1和事务2获取不同行的锁是可以成功的。然而现在我们发现获取name为chenwei的排它锁也获取不到了,这是为什么?我们解释一下:

InnoDB的行锁是通过给索引项加锁来实现的,而不是给表的行记录加锁实现的

而我们用name作为过滤条件没有用到索引,自然就不会使用行锁,而是使用表锁。这就意味着只有通过索引检索数据,InnoDB才使用行级锁,否则InnoDB都将使用表锁!!!

我们给name字段加上索引:

MySQL的表级锁,行级锁,排它锁和共享锁

然后再做刚才的操作:

MySQL的表级锁,行级锁,排它锁和共享锁

我们发现,给name加上索引后,两个事务可以获取到不同行的排它锁(for update),再一次证明了InnoDB的行锁是加在索引项上的。

MySQL的表级锁,行级锁,排它锁和共享锁

因为现在name走的是索引, 通过zhangsan在辅助索引树上找到它所在行记录的id是7,然后到主键索引树上,获取对应行记录的排他锁(个人猜测应该是辅助索引树和主键索引树相应的记录都加了锁)

三、串行化隔离级别测试

串行化所有事务用的都是共享锁或者排它锁,不需用手动添加。select获取的是共享锁,insert、delete和update获取的都是排它锁。

设置串行化隔离级别:

MySQL的表级锁,行级锁,排它锁和共享锁

两个事务可以同时获取共享锁(SS共存:

MySQL的表级锁,行级锁,排它锁和共享锁

现在让事务2插入数据;

MySQL的表级锁,行级锁,排它锁和共享锁

由于Insert需要加排它锁,但是由于事务1已经对整张表加了共享锁,事务2无法再对表成功加锁(sx不共存)

rollback一下,把所有获取锁的状态都回退掉:

MySQL的表级锁,行级锁,排它锁和共享锁

开启两个事务:

MySQL的表级锁,行级锁,排它锁和共享锁

因为我们给name加上了索引,以上的select相当于给name为zhangsan的数据加上了行共享锁

事务2update;

MySQL的表级锁,行级锁,排它锁和共享锁

事务2不能update,因为此时已经被事务1的共享锁锁住了整个表

事务2在辅助索引树上找zhangsan,找到对应的主键值,然后去主键索引树找到相应的记录,但是发现这行记录已经被共享锁锁住了,事务2可以获取共享锁,但是不能获取排他锁

MySQL的表级锁,行级锁,排它锁和共享锁

我们再用主键索引试试id能不能update

MySQL的表级锁,行级锁,排它锁和共享锁

依然阻塞住了,虽然我们where后面的字段现在使用的id而不是name,但是name也是通过辅助索引树找到对应的主键,再到主键索引树上找相应的记录,而主键索引树上的记录加了锁

我们update id=8的数据,成功了。因为我们select的时候,只是给id=7的数据加上了行锁,我们操作id=8的数据当然可以成功

MySQL的表级锁,行级锁,排它锁和共享锁

有索引,则使用行锁;没有索引,则使用表锁。

表级锁还是行级锁说的是锁的粒度,共享锁和排他锁说的是锁的性质,不管是表锁还是行锁,都有共享锁和排他锁的区分。

串行化玩的就是排它锁和共享锁,在可重复读级别下,不手动加锁的话,用的就是MVCC机制,实际上并没有用到锁,我们也可以手动加锁。InnoDB如果不创建索引的话,用的是表锁,如果查询的时候用到了索引项,它用的就是行锁了,行锁是给索引加锁,而不是单纯给一行数据加锁。

到此这篇关于MySQL的表级锁,行级锁,排它锁和共享锁的文章就介绍到这了,更多相关MySQL锁内容请搜索三水点靠木以前的文章或继续浏览下面的相关文章希望大家以后多多支持三水点靠木!

MySQL 相关文章推荐
MySQL update set 和 and的区别
May 08 MySQL
MySQL中distinct与group by之间的性能进行比较
May 26 MySQL
Mysql 如何查询时间段交集
Jun 08 MySQL
浅谈MySQL 亿级数据分页的优化
Jun 15 MySQL
新手入门Mysql--概念
Jun 18 MySQL
MySQL连接控制插件介绍
Sep 25 MySQL
浅谈MySQL表空间回收的正确姿势
Oct 05 MySQL
如何避免mysql启动时错误及sock文件作用分析
Jan 22 MySQL
weblogic服务建立数据源连接测试更新mysql驱动包的问题及解决方法
Jan 22 MySQL
利用JuiceFS使MySQL 备份验证性能提升 10 倍
Mar 17 MySQL
MySQL主从切换的超详细步骤
Jun 28 MySQL
MySQL的意向共享锁、意向排它锁和死锁
Jul 15 MySQL
MySQL事务的隔离级别详情
Jul 15 #MySQL
MySQL事务的ACID特性以及并发问题方案
Jul 15 #MySQL
MySQL的意向共享锁、意向排它锁和死锁
Jul 15 #MySQL
Mysql数据库group by原理详解
delete in子查询不走索引问题分析
Jul 07 #MySQL
MySQL提升大量数据查询效率的优化神器
mysql查看表结构的三种方法总结
Jul 07 #MySQL
You might like
Yii2简单实现多语言配置的方法
2016/07/23 PHP
php及codeigniter使用session-cookie的方法(详解)
2017/04/06 PHP
PHP封装的验证码工具类定义与用法示例
2018/08/22 PHP
js中匿名函数的N种写法
2010/09/08 Javascript
获取当前点击按钮的id用this.id实现
2014/03/17 Javascript
jquery队列queue与原生模仿其实现方法分享
2014/03/25 Javascript
浅谈Sizzle的“编译原理”
2015/04/14 Javascript
JS与jQuery遍历Table所有单元格内容的方法
2015/12/07 Javascript
微信小程序中为什么使用var that=this
2019/08/27 Javascript
解决layui页面按钮点击无反应,也不报错的问题
2019/09/29 Javascript
JS操作Fckeditor的一些常用方法(获取、插入等)
2020/02/19 Javascript
在Webpack中用url-loader处理图片和字体的问题
2020/04/28 Javascript
Vue使用预渲染代替SSR的方法
2020/07/02 Javascript
实例讲解JavaScript 计时事件
2020/07/04 Javascript
详解阿里Node.js技术文档之process模块学习指南
2021/01/04 Javascript
在Python的Flask框架中实现单元测试的教程
2015/04/20 Python
Python文本特征抽取与向量化算法学习
2017/12/22 Python
pandas.dataframe按行索引表达式选取方法
2018/10/30 Python
Pandas Shift函数的基础入门学习笔记
2018/11/16 Python
PyTorch搭建多项式回归模型(三)
2019/05/22 Python
Jacobi迭代算法的Python实现详解
2019/06/29 Python
Python第三方包之DingDingBot钉钉机器人
2020/04/09 Python
使用PyQt5实现图片查看器的示例代码
2020/04/21 Python
matplotlib常见函数之plt.rcParams、matshow的使用(坐标轴设置)
2021/01/05 Python
利用CSS3动画实现圆圈由小变大向外扩散的效果实例
2018/09/10 HTML / CSS
荷兰游戏商店:Allyouplay
2019/03/16 全球购物
STP的判定过程
2012/10/01 面试题
有模特经验的简历自我评价
2013/09/19 职场文书
计算机专业推荐信范文
2013/11/20 职场文书
中专三年学习的个人自我评价
2013/12/12 职场文书
抄作业检讨书
2014/02/17 职场文书
2014庆六一活动方案
2014/03/02 职场文书
升学宴答谢词
2015/01/05 职场文书
社区五一劳动节活动总结
2015/02/09 职场文书
如何在Mac上通过docker配置PHP开发环境
2021/05/29 PHP
vue-cli3.x配置全局的scss的时候报错问题及解决
2022/04/30 Vue.js