MySQL的意向共享锁、意向排它锁和死锁


Posted in MySQL onJuly 15, 2022

一、InnoDB的表级锁

在绝大多数情况下应该使用行锁,因为事务和行锁往往是选择InnoDB的理由,但个别情况下也使用表级锁。

事务需要更新大部分或全部数据,表又比较大,如果使用默认的行锁,不仅这个事务执行效率低,而且可能造成其他事务长时间等待和锁冲突事务涉及多个表,比较复杂,很可能引起死锁,造成大量事务回滚

我们希望获取表锁时,执行以下命令:

MySQL的意向共享锁、意向排它锁和死锁

在使用表锁的时候,涉及的效率问题:
要获取一张表的共享锁S或者排它锁X,首先要确定这张表没有被其它事务获取过X锁,这张表里面的数据没有被其它事务获取过行锁X锁。

假如我们这张表有一千万个数据,怎么判断这一千万个数据哪些行有X锁?
如果想获得表的S锁,就需要判断表里面哪些行有过X锁,如果有一些行有过X锁,那么就不能获取这张表的S锁或者X锁。除了挨个检查,没有更好的办法,这就导致效率低下的问题

由于需要加表锁而去挨个遍历数据,确定是否有某些数据被加了行锁,而导致的效率低下问题。我们这里学习的意向共享锁和意向排他锁就可以解决,当要获取表的X锁是,不需要再检查表中的哪些行锁被(X或者S)占用,只需要快速检查IX和IS锁即可

二、意向共享锁和意向排它锁

  • 意向共享锁(IS锁):事务计划给记录加行共享锁,事务在给一行记录加共享锁前,必须先取得该表的IS锁
  • 意向排他锁(IX锁):事务计划给记录加行排他锁,事务在给一行记录加排他锁前,必须先取得该表的IX锁

MySQL的意向共享锁、意向排它锁和死锁

  • 在加行锁之前,由InnoDB存储引擎加上表的IS或IX锁
  • 意向锁之间都是兼容的,不会产生冲突,主要是为了辅助其他的在获取表锁的时候加快效率
  • 意向锁存在的意义是为了更高效的获取表锁(表格中的X和S指的是表锁,不是行锁!)
  • 意向锁是表级锁协调表锁和行锁的共存关系。主要目的是显示事务正在锁定某行或者试图锁定某行。

分析事务1获取行X锁和事务2获取表S锁:

首先事务1需要给表的第10行数据加X锁,于是InnoDB存储引擎自动给整张表加上了IX锁。当事务2再想获取整张表的S锁时,看到这张表已经有别的事务获取了IX锁了,就说明这张表肯定有某些数据被加上了X锁,这就导致事务2不能给整张表加S锁了。此时事务2只能等待,无法成功获取表S锁

三、死锁

1. 数据库中的死锁

MyISAM表锁时deadlock free的,这是因为MyISAM 不支持事务,只支持表锁,总是一次获得所需的全部锁,要么全部满足,要么等待,因此不会出现死锁。但在InnoDB中,除单个SQL组成的事务外,锁是逐步获得的,即锁的粒度比较小,这就决定了在InnoDB中发生死锁是可能的。当然如果处理多张表,还是有可能出现死锁的。

死锁问题一般都是我们自己造成的,和多线程编程的死锁情况相似,大部分都是由于我们多个线程在获取多个锁资源的时候,获取的顺序不同而导致的死锁问题。因此我们应用在对数据库的多个表做更新的时候,不同的代码段,应对这些表按相同的顺序进行更新操作,以防止锁冲突导致死锁问题。

2. 死锁场景以及解决方法

死锁出现的场景如下:

事务1成功获取行锁1
事务2成功获取行锁2

事务1无法获取行锁2,被阻塞的同时没有办法执行commit/rollback,无法释放行锁1
事务2无法获取行锁1,被阻塞的同时没有办法执行commit/rollback,无法释放行锁2

MySQL的意向共享锁、意向排它锁和死锁

所有的事务都阻塞了,相当于进程内的所有线程都阻塞了,造成了死锁问题。

解决死锁的方法:多个事务/线程获取多个相同资源锁的时候,应该按照同样的顺序,获取资源的锁。

事务被阻塞或者死锁了,mysqld(MySQL Server守护进程)设置有事务阻塞的超时时间,事务不会阻塞很长时间,超时后事务处理失败,自动释放当前占有的锁。

3. 操作

设置手动提交和可重复读隔离级别并开启事务

MySQL的意向共享锁、意向排它锁和死锁

查询一下表数据,在可重复读隔离级别使用的是MVCC提供的快照读,并没有加锁

MySQL的意向共享锁、意向排它锁和死锁

事务1获取id=7的排他锁,事务2获取id=8的排他锁

MySQL的意向共享锁、意向排它锁和死锁

事务1再获取id=8的排它锁,发生阻塞

MySQL的意向共享锁、意向排它锁和死锁

事务2再获取id=7的排它锁,发生阻塞

MySQL的意向共享锁、意向排它锁和死锁

此时由于MySQL Server检测到发生了死锁,于是解除事务1的阻塞,进行事务1的rollback,释放其占有的行锁,于是事务2成功获取id=7的排它锁

三、锁的优化建议

  • 在能正确完成业务的前提下,为确保效率,尽量使用较低的隔离级别(必须避免脏读)
  • 设计合理的索引并尽量使用索引访问数据,使加锁更准确,减少锁冲突的机会,提高并发能力
  • 选择合理的事务大小,小事务发生锁冲突的概率小(事务越大,包含的SQL越多,可能包含更多的表资源和行资源的锁,增大了锁冲突的概率)不同的程序访问一组表时,应尽量约定以相同的顺序访问各表,对一个表而言,尽可能以固定的顺序存取表中的行。这样可以大大减少死锁的机会
  • 尽量用相等条件访问数据,这样可以避免间隙锁对并发插入的影响(其实等值查询也会加间隙锁)不要申请超过实际需要的锁级别
  • 除非必须,查询时不要显示加锁(在已提交读和可重复读隔离级别,MVCC提供了读取机制,不需要手动加锁)

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

MySQL 相关文章推荐
MySQL 重命名表的操作方法及注意事项
May 21 MySQL
MySQL触发器的使用
May 24 MySQL
MySQL 查询速度慢的原因
May 25 MySQL
浅谈MySQL之select优化方案
Aug 07 MySQL
Mysql binlog日志文件过大的解决
Oct 05 MySQL
浅谈MySQL表空间回收的正确姿势
Oct 05 MySQL
SQL优化老出错,那是你没弄明白MySQL解释计划用法
Nov 27 MySQL
Mysql存储过程、触发器、事件调度器使用入门指南
Jan 22 MySQL
MySQL 分区表中分区键为什么必须是主键的一部分
Mar 17 MySQL
MySQL数据库优化之通过索引解决SQL性能问题
Apr 10 MySQL
mysql序号rownum行号实现方式
Dec 24 MySQL
Mysql数据库group by原理详解
delete in子查询不走索引问题分析
Jul 07 #MySQL
MySQL提升大量数据查询效率的优化神器
mysql查看表结构的三种方法总结
Jul 07 #MySQL
MySQL中正则表达式(REGEXP)使用详解
MySQL实现字段分割一行转多行的示例代码
MySQL控制流函数(-if ,elseif,else,case...when)
Jul 07 #MySQL
You might like
php学习笔记 面向对象中[接口]与[多态性]的应用
2011/06/16 PHP
php安全之直接用$获取值而不$_GET 字符转义
2012/06/03 PHP
PHP MPDF中文乱码的解决方式
2015/12/08 PHP
PHP使用stream_context_create()模拟POST/GET请求的方法
2016/04/02 PHP
CI框架AR操作(数组形式)实现插入多条sql数据的方法
2016/05/18 PHP
论坛特效代码收集(落伍转发-不错)
2006/12/02 Javascript
Prototype Hash对象 学习
2009/07/19 Javascript
关于js中window.location.href,location.href,parent.location.href,top.location.href的用法与区别
2010/10/18 Javascript
基于jquery的从一个页面跳转到另一个页面的指定位置的实现代码(带平滑移动的效果)
2011/05/24 Javascript
JS图片无缝、平滑滚动代码
2014/03/11 Javascript
javascript检测浏览器的缩放状态实现代码
2014/09/28 Javascript
JS获取鼠标坐标位置实例分析
2016/01/20 Javascript
jQuery实现根据滚动条位置加载相应内容功能
2016/07/18 Javascript
js动态添加的DIV中的onclick事件简单实例
2016/07/25 Javascript
JSP防止网页刷新重复提交数据的几种方法
2016/11/19 Javascript
用jQuery.ajaxSetup实现对请求和响应数据的过滤
2016/12/20 Javascript
详解webpack 多入口配置
2017/06/16 Javascript
关于定制FileField中的上传文件名称问题
2017/08/22 Javascript
chorme 浏览器记住密码后input黄色背景处理方法(两种)
2017/11/22 Javascript
详解vue-meta如何让你更优雅的管理头部标签
2018/01/18 Javascript
纯JS实现出生日期[年月日]下拉菜单效果
2018/06/01 Javascript
详解nuxt 微信公众号支付遇到的问题与解决
2019/08/26 Javascript
[52:09]2014 DOTA2华西杯精英邀请赛 5 25 NewBee VS DK第二场
2014/05/26 DOTA
测试、预发布后用python检测网页是否有日常链接
2014/06/03 Python
Python爬虫利用cookie实现模拟登陆实例详解
2017/01/12 Python
Python的装饰器使用详解
2017/06/26 Python
Python中的X[:,0]、X[:,1]、X[:,:,0]、X[:,:,1]、X[:,m:n]和X[:,:,m:n]
2020/02/13 Python
python利用opencv实现SIFT特征提取与匹配
2020/03/05 Python
浅析Django 接收所有文件,前端展示文件(包括视频,文件,图片)ajax请求
2020/03/09 Python
Python约瑟夫生者死者小游戏实例讲解
2021/01/04 Python
HTML5使用DOM进行自定义控制示例代码
2013/06/08 HTML / CSS
HTML5 文件域+FileReader 分段读取文件并上传到服务器
2017/10/23 HTML / CSS
英国虚拟主机服务商:eUKhost
2016/08/16 全球购物
LightInTheBox西班牙站点:全球商品在线采购
2016/09/22 全球购物
房地产还款计划书
2014/01/10 职场文书
Java实现扫雷游戏详细代码讲解
2022/05/25 Java/Android