这篇文章给大家分享的是有关如何使用不同的索引更新解决MySQL死锁的内容。小编觉得挺实用的,因此分享给大家做个参考,一起跟随小编过来看看吧。

示例如下

CREATETABLE`t3`(`id`int(11)NOTNULLAUTO_INCREMENT,`a`varchar(5),`b`varchar(5),PRIMARYKEY(`id`),UNIQUEKEY`uk_a`(`a`),KEY`idx_b`(`b`))INSERTINTO`t3`(`id`,`a`,`b`)VALUES(1,'1','2');#sql语句如下#事务1:t1updatet3setb=''wherea="1";#事务2:t2updatet3setb=''whereb="2";

两条语句造成死锁的情况用手动的方式比较难复现,我们先来分析一下加锁的过程

第一条语句(通过唯一索引去更新记录)

updatet3setb=''wherea="1";

整理一下,加了3个X锁,顺序分别是

序号索引锁类型1uk_aX2PRIMARYX3idx_bX

第二条语句



updatet3setb=''whereb="2";

整理一下,加了 3 个 X 锁,顺序分别是

序号索引锁类型1idx_bX2PRIMARYX3idx_bX

两条语句从加锁顺序看起来就已经有构成死锁的条件了

手动是比较难模拟的,写个代码并发的去同时执行那两条 SQL 语句,马上就出现死锁了

------------------------LATESTDETECTEDDEADLOCK------------------------18110212:45:05***(1)TRANSACTION:TRANSACTION50AF,ACTIVE0secstartingindexreadmysqltablesinuse1,locked1LOCKWAIT3lockstruct(s),heapsize376,2rowlock(s)MySQLthreadid34,OSthreadhandle0x70000d842000,queryid549localhost127.0.0.1rootSearchingrowsforupdateupdatet3setb=''whereb="2"***(1)WAITINGFORTHISLOCKTOBEGRANTED:RECORDLOCKSspaceid67pageno3nbits72index`PRIMARY`oftable`d1`.`t3`trxid50AFlock_modeXlocksrecbutnotgapwaitingRecordlock,heapno2PHYSICALRECORD:n_fields5;compactformat;infobits00:len4;hex80000001;asc;;1:len6;hex0000000050ae;ascP;;2:len7;hex03000001341003;asc4;;3:len1;hex31;asc1;;4:len0;hex;asc;;***(2)TRANSACTION:TRANSACTION50AE,ACTIVE0secupdatingordeletingmysqltablesinuse1,locked14lockstruct(s),heapsize1248,3rowlock(s),undologentries1MySQLthreadid35,OSthreadhandle0x70000d885000,queryid548localhost127.0.0.1rootUpdatingupdatet3setb=''wherea="1"***(2)HOLDSTHELOCK(S):RECORDLOCKSspaceid67pageno3nbits72index`PRIMARY`oftable`d1`.`t3`trxid50AElock_modeXlocksrecbutnotgapRecordlock,heapno2PHYSICALRECORD:n_fields5;compactformat;infobits00:len4;hex80000001;asc;;1:len6;hex0000000050ae;ascP;;2:len7;hex03000001341003;asc4;;3:len1;hex31;asc1;;4:len0;hex;asc;;***(2)WAITINGFORTHISLOCKTOBEGRANTED:RECORDLOCKSspaceid67pageno5nbits72index`idx_b`oftable`d1`.`t3`trxid50AElock_modeXlocksrecbutnotgapwaitingRecordlock,heapno2PHYSICALRECORD:n_fields2;compactformat;infobits00:len1;hex32;asc2;;1:len4;hex80000001;asc;;***WEROLLBACKTRANSACTION(1)

分析一下死锁日志

***(1)WAITINGFORTHISLOCKTOBEGRANTED:RECORDLOCKSspaceid67pageno3nbits72indexPRIMARYoftabled1.t3trxid50AFlock_modeXlocksrecbutnotgapwaiting

事务2:想获取主键索引的 X 锁

***(2)HOLDSTHELOCK(S):RECORDLOCKSspaceid67pageno3nbits72indexPRIMARYoftabled1.t3trxid50AElock_modeXlocksrecbutnotgap

事务1:持有主键索引的 X 锁

***(2)WAITINGFORTHISLOCKTOBEGRANTED:RECORDLOCKSspaceid67pageno5nbits72indexidx_boftabled1.t3trxid50AElock_modeXlocksrecbutnotgapwaiting

事务1:想获取普通索引 idx_b 的 X 锁

与我们分析的完全一致,也与线上的死锁日志一模一样

感谢各位的阅读!关于“如何使用不同的索引更新解决MySQL死锁”这篇文章就分享到这里了,希望以上内容可以对大家有一定的帮助,让大家可以学到更多知识,如果觉得文章不错,可以把它分享出去让更多的人看到吧!