位置: 编程技术 - 正文

Mysql 数据库死锁过程分析(select for update)(mysql数据库死锁产生的原因及解决方案)

编辑:rootadmin

推荐整理分享Mysql 数据库死锁过程分析(select for update)(mysql数据库死锁产生的原因及解决方案),希望有所帮助,仅作参考,欢迎阅读内容。

文章相关热门搜索词:mysql数据库死锁处理方法有哪些,mysql数据库死锁处理方法有哪些,mysql数据库死锁排查,mysql数据库死锁排查,mysql数据库死锁产生的原因及解决方案,mysql数据库死锁产生的原因,mysql数据库死锁捕获什么异常,mysql数据库死锁排查,内容如对您有帮助,希望把文章链接给更多的朋友!

近期有一个业务需求,多台机器需要同时从Mysql一个表里查询数据并做后续业务逻辑,为了防止多台机器同时拿到一样的数据,每台机器需要在获取时锁住获取数据的数据段,保证多台机器不拿到相同的数据。

我们Mysql的存储引擎是innodb,支持行锁。解决同时拿数据的方法有很多,为了更加简单,不增加其他表和服务的情况下,我们考虑采用select... for update的方式,这样X锁锁住查询的数据段,表里其他数据没有锁,其他业务逻辑还是可以操作。

这样一台服务器比如select .. for update limit 0,时,其他服务器执行同样sql语句会自动等待释放锁,等待前一台服务器锁释放后,该台服务器就能查询下一个条数据。如果要求更智能,oracle支持for update skip locked跳过锁区域,这样能不等待马上查询没有被锁住的下一个条记录。

下面说下mysql for update导致的死锁。

经过分析,mysql的innodb存储引擎实务锁虽然是锁行,但它内部是锁索引的,根据where条件和select的值是否只有主键或非主键索引来判断怎么锁,比如只有主键,则锁主键索引,如果只有非主键,则锁非主键索引,如果主键非主键都有,则内部会按照顺序锁。但同样的select .. for update语句怎么就死锁了呢?同样的sql语句查询条件和结果顺序都一致,按理不会导致一个锁了主键索引,等待锁非主键索引,另外一个锁了非主键索引,等待主键索引导致的死锁。

最后经过分析,我们项目里发现是for update的sql语句,和另外一个update非select数据的sql语句导致的死锁。

比如有条数据,select .. for update查询第-条数据,update在更新1-条数据,按照innodb存储引擎的行锁原理,应该不会导致不同行的锁导致的互相等待。开始以为是行锁在数据量较大情况下,会锁数据块。导致一个段的数据被锁住,但经过大量数据测试,发现感觉把整个表都锁住了,但实际不是。

下面举几个例子说明:

数据从id =的数据开始,IsSuccess和GetTime字段都为0,现在如果数据的IsSuccess为1了。执行下面两条sql.

  第一条sql语句先不commit,则第二条sql语句将只能等待,因此第二条sql语句把IsSuccess修改为0,IsSuccess非主键索引锁了值为0的索引数据,第二条sql语句将无法把数据更新到被锁的行里。

再执行下面的sql语句

  这样第二条sql语句将可以执行。因为IsSuccess=2的索引段没有被锁。

上面的例子知道了锁索引段后还比较容易看懂,下面就奇葩一点:

先把id =数据的GetTime修改为1,IsSuccess=0,然后一次执行sql:

第1个sql先不commit,按照道理只会锁这行记录,第二个sql执行,按照道理只能查询从记录的条记录,但第二个sql语句会阻塞等待。

原因是第一个sql语句还没有commit也没有rollback,因此它先锁主键索引,再锁IsSuccess的非主键索引,第二个sql语句由于where里要判断IsSuccess字段的值,由于这条数据以前的IsSuccess是0,现在更新为1还不确定,可能会回滚,因此sql2需要等待确定这条数据的IsSuccess是否被修改。sql2的sql语句因为判断了GetTime<1,实际这条记录已经不满足了,但按照锁索引的原理,所以sql2语句会被阻塞。

Mysql 数据库死锁过程分析(select for update)(mysql数据库死锁产生的原因及解决方案)

因此如果根据业务场景,可以把sql2语句的IsSuccess条件取消掉,并且这里GetTime查询条件由GetTime<1修改为GetTime=0,这样即可不阻塞直接查询出来。

GetTime用范围查询导致的锁影响经过分析,还不是间隙锁的问题,感觉应该是用范围作为条件,所有从第0行开始的所有查找范围都会被锁住。 比如这里更新会被阻塞,但更新不会被阻塞。

我们项目出现死锁,就是这个原理,一条sql语句先锁主键索引,再锁非主键索引;另外一条sql语句先锁非主键索引,再锁主键索引。虽然两个sql语句期望锁的数据行不一样,但两个sql语句查询或更新的条件或结果字段如果有相同列,则可能会导致互相等待对方锁,2个sql语句即引起了死锁。

个人总结一下innodb存储引擎下的锁的分析,可能会有问题:

1、更新或查询for update的时候,会在where条件中开始为每个字段判断是否有锁,如果有锁就会等待,因为如果有锁,那这个字段的值不确定,只能等待锁commit或rollback后数据确定后再查询。

2、另外还和order by有关系,因为可能前面数据有锁,但从后面查询一个范围就可以查询。

3、另外limit也有关系,比如limit ,从第条记录取行数据,但第一行数据如果被锁,因为不确定回滚还是提交,也会锁等待。

ps:mysql使用kill命令解决死锁问题,杀死某条正在执行的sql语句

使用mysql运行某些语句时,会因数据量太大而导致死锁,没有反映。这个时候,就需要kill掉某个正在消耗资源的query语句即可, KILL命令的语法格式如下:

每个与mysqld的连接都在一个独立的线程里运行,您可以使用SHOW PROCESSLIST语句查看哪些线程正在运行,并使用KILL thread_id语句终止一个线程。

KILL允许自选的CONNECTION或QUERY修改符:KILL CONNECTION与不含修改符的KILL一样:它会终止与给定的thread_id有关的连接。KILL QUERY会终止连接当前正在执行的语句,但是会保持连接的原状。

如果您拥有PROCESS权限,则您可以查看所有线程。如果您拥有超级管理员权限,您可以终止所有线程和语句。否则,您只能查看和终止您自己的线程和语句。您也可以使用mysqladmin processlist和mysqladmin kill命令来检查和终止线程。

首先登录mysql,然后使用: show processlist; 查看当前mysql中各个线程状态。

以上显示出当前正在执行的sql语句列表,找到消耗资源最大的那条语句对应的id.

然后运行kill命令,命令格式如下:

kill id;-- 示例: kill

杀掉即可。

Java的Struts框架中的主题模板和国际化设置 主题模板如果不指定一个主题,然后Struts2中会使用默认的XHTML主题。例如Struts2中选择标签:s:textfieldname="name"label="Name"/生成HTML标记:trtdclass="tdLabel"labelf

解决MySQL客户端输出窗口显示中文乱码问题的办法 最近发现,在MySQL的dos客户端输出窗口中查询表中的数据时,表中的中文数据都显示成乱码,如下图所示:上网查了一下原因:之所以会显示乱码,就是

安装Mysql时出现错误及解决办法 因为一时手痒痒更新了一下驱动,结果导致无线网卡出了问题。然而就算是从官网上下载了驱动各种折腾也没有弄好,心里特别堵。无奈只有重装系统

标签: mysql数据库死锁产生的原因及解决方案

本文链接地址:https://www.jiuchutong.com/biancheng/312397.html 转载请保留说明!

上一篇:MySQL如何清空慢查询文件(mysql清空缓存)

下一篇:Java的Struts框架中的主题模板和国际化设置(javagui框架)

  • 出口退税的账怎么做
  • 服务费发票的税率是多少
  • 查找出资产负债表的软件
  • 增值税季报还是月报
  • 什么时候用以前年度损益调整什么时候用年初未分配利润
  • 应交税费销项税在借方还是贷方
  • 已付的账款叫什么
  • 外贸企业当月没交税
  • 企业年报 工商
  • 出售无形资产净收益是收入吗
  • 凭证审核签字操作只能
  • 现金流管理模式
  • 长期股权投资的初始计量
  • 本月完工产品的会计分录
  • 待处置资产损溢在什么科目
  • 制造业贷款用途
  • 开红字增值税专用发票步骤
  • 农产品收购发票怎么抵扣
  • 远程认证是什么意思
  • 农民专业合作社属于什么企业类型
  • 购进材料再销售怎么会计分录
  • 审计报告格式与范文怎么写?
  • 社保和医保是分开到账吗
  • 负债转为投资 资本增加吗
  • 腾讯电脑管家中有没有红色警戒下载玩
  • 王者荣耀百里守约是男是女
  • 已认证的发票退税怎么退
  • 中介公司收取中介费过高违法吗
  • 进项发票没认证可以开红字申请单吗
  • 王者荣耀中刘邦技能解析以及如何连招
  • 进口应税消费品的组成计税价格为
  • 如何给网页添加水印
  • linux系统之间拷贝文件命令
  • kb4499164安装失败怎么办
  • php mb_convert_encoding
  • php字符串变量
  • 详解HTTP Cookie状态管理机制
  • 出现质量问题赔偿标准
  • 收到退回的以前年度的劳务费怎么入账
  • uni-app--》uni-app的生命周期讲解
  • openai.error.AuthenticationError: No API key provided.
  • vue的内置组件
  • 小微企业的增值税税收优惠政策
  • 汇算清缴补交所得税怎么做凭证
  • 补交去年增值税怎么做账
  • 免征的教育费附加怎么做账
  • 上月附加税计提多了怎么办
  • 零申报企业所得税季度申报表怎么填写
  • python多进程间通信
  • 金税盘减免税款月末如何结转
  • 销售旧货的增值税是销项税吗
  • 啤酒的消费税
  • 购买财务软件报税流程
  • 退回的以前年度的附加怎么申报
  • 清算中的资产损失
  • 专项应付款的核算
  • 哪些原始凭证要盖章
  • 待抵扣进项税计入其他应付账款吗
  • 纸质承兑汇票怎么兑现步骤
  • 有留抵税额的会计处理
  • 融资租赁可以折旧吗
  • 企业所得税收入是含税还是不含税
  • 老毛桃u盘启动制作工具如何把原来的win7改xp系统图文教程
  • windows dns server
  • wp7.8升级wp8
  • linux使用场合
  • psoft1.exe - psoft1是什么进程 有什么作用
  • win8适合打游戏吗
  • centos创建一个文件
  • win10注册不了账号
  • linux小技巧
  • 机械革命系统重装官方教程
  • three.js入门教程(合集)
  • jquery json对象
  • $jquery
  • bootstrap基础教程
  • jquery的选择器都有哪些
  • 货车可申请停保吗
  • 地税局和税务局有什么区别
  • 税务申报网上申报
  • 免责声明:网站部分图片文字素材来源于网络,如有侵权,请及时告知,我们会第一时间删除,谢谢! 邮箱:opceo@qq.com

    鄂ICP备2023003026号

    网站地图: 企业信息 工商信息 财税知识 网络常识 编程技术

    友情链接: 武汉网站建设