详解MySQL(InnoDB)如何处理死锁 锁是需要事务结束后才释放的 。
一个是 MVCC,一个是两阶段锁协议 。
为什么要并发控制呢?是因为多个用户同时操作 MySQL 的时候,为了提高并发性能并且要求如同多个用户的请求过来之后如同串行执行的一样(为了解决脏读、不可重复读、幻读)
官方定义:
两阶段锁协议是指所有事务必须分两个阶段对数据加锁和解锁,在对任何数据进行读、写操作之前,事务首先要获得对该数据的封锁;在释放一个封锁之后 , 事务不再申请和获得任何其他封锁 。
对应到 MySQL 上分为两个阶段:
但是两阶段锁协议不要求事务必须一次将所有需要使用的数据加锁(innodb在需要的索引列数据才锁行),并且在加锁阶段没有顺序要求,所以这种并发控制方式会形成死锁 。
MySQL有两种死锁处理方式:
死锁检测 (默认开启)
死锁检测的原理是构建一个以事务为顶点、锁为边的有向图,判断有向图是否存在环,存在即有死锁 。
回滚
检测到死锁之后,选择插入更新或者删除的行数最少的事务回滚,基于 INFORMATION_SCHEMA.INNODB_TRX 表中的 trx_weight 字段来判断 。
收集死锁信息:
减少死锁:
死锁解决:
MySQL数据库表被锁、解锁,删除事务在程序员的职业生涯中,总会遇到数据库表被锁的情况,前些天就又撞见一次 。由于业务突发需求,各个部门都在批量操作、导出数据,而数据库又未做读写分离,结果就是:数据库的某张表被锁了!
用户反馈系统部分功能无法使用,紧急排查,定位是数据库表被锁 , 然后进行紧急处理 。这篇文章给大家讲讲遇到类似紧急状况的排查及解决过程,建议点赞收藏,以备不时之需 。
用户反馈某功能页面报502错误,于是第一时间看服务是否正常,数据库是否正常 。在控制台看到数据库CPU飙升,堆积大量未提交事务,部分事务已经阻塞了很长时间,基本定位是数据库层出现问题了 。
查看阻塞事务列表,发现其中有锁表现象,本想利用控制台直接结束掉阻塞的事务,但控制台账号权限有限 , 于是通过客户端登录对应账号将锁表事务kill掉,才避免了情况恶化 。
下面就聊聊,如果当突然面对类似的情况,我们该如何紧急响应?
想象一个场景,当然也是软件工程师职业生涯中会遇到的一种场景:原本运行正常的程序,某一天突然数据库的表被锁了,业务无法正常运转,那么我们该如何快速定位是哪个事务锁了表,如何结束对应的事物?
首先最简单粗暴的方式就是:重启MySQL 。对的,网管解决问题的神器——“重启” 。至于后果如何 , 你能不能跑了 , 要你自己三思而后行了!
重启是可以解决表被锁的问题的,但针对线上业务很显然不太具有可行性 。
下面来看看不用跑路的解决方案:
遇到数据库阻塞问题,首先要查询一下表是否在使用 。
如果查询结果为空,那么说明表没在使用,说明不是锁表的问题 。
如果查询结果不为空,比如出现如下结果:
则说明表(test)正在被使用,此时需要进一步排查 。
查看数据库当前的进程,看看是否有慢SQL或被阻塞的线程 。
执行命令:
该命令只显示当前用户正在运行的线程,当然,如果是root用户是能看到所有的 。
在上述实践中,阿里云控制台之所以能够查看到所有的线程,猜测应该使用的就是root用户,而笔者去kill的时候,无法kill掉,是因为登录的用户非root的数据库账号,无法操作另外一个用户的线程 。
推荐阅读
- 触电直播新闻平台,触电新闻直播间
- 影片编辑器安卓,影音编辑器
- linux如何结束命令,linux 结束命令
- 手机屏幕怎么弄不会容易黑,手机怎么屏幕不会黑屏
- linux下sz命令 linux中sz
- hbase的基本sql使用,hbase select
- 休闲经营类游戏名字,好玩的休闲经营类游戏
- window安装flutter,window安装mysql8
- php发送二进制数据流 php 二进制流