MySQL Innodb表导致死锁日志情况分析与归纳
案例描述 死锁日志打印出的时间点表明,语句(1)运行过程中,当语句(2)开始运行时,发生了死锁。 根据死锁记录的结果,可以看出确实是这两个语句发生了死锁,且锁冲突发生在主键索引上。那么,为什么两个sql语句会存在锁冲突呢?冲突为什么会在主键索引上呢?语句(2)得到了主键索引锁,为什么还会再次申请锁呢? 锁冲突分析 2.1 innodb的事务与行锁机制 MySQL的事务支持不是绑定在MySQL服务器本身,而是与存储引擎相关,MyISAM不支持事务、采用的是表级锁,而InnoDB支持ACID事务、 行级锁、并发。MySQL默认的行为是在每条SQL语句执行后执行一个COMMIT语句,从而有效的将每条语句作为一个单独的事务来处理。 2.2 两语句加锁情况 在innodb默认的事务隔离级别下,普通的SELECT是不需要加行锁的,但LOCK IN SHARE MODE、FOR UPDATE及高串行化级别中的SELECT都要加锁。有一个例外,此案例中,语句(1)insert into teamUser_20110121 select * from teamUser会对表teamUser_20110121(ENGINE= MyISAM)加表锁,并对teamUser表所有行的主键索引(即聚簇索引)加共享锁。默认对其使用主键索引。 而语句(2)DELETE FROM teamUser WHERE teamId=$teamId AND titleWeight<32768 AND joinTime<'$daysago_1week'为删除操作,会对选中行的主键索引加排他锁。由于此语句还使用了非聚簇索引KEY `k_teamid_titleWeight_score` (`teamId`,`score`)的前缀索引,于是,还会对相关行的此非聚簇索引加排他锁。 2.3 锁冲突的产生 由于共享锁与排他锁是互斥的,当一方拥有了某行记录的排他锁后,另一方就不能其拥有共享锁,同样,一方拥有了其共享锁后,另一方也无法得到其排他锁。所 以,当语句(1)、(2)同时运行时,相当于两个事务会同时申请某相同记录行的锁资源,于是会产生锁冲突。由于两个事务都会申请主键索引,锁冲突只会发生 在主键索引上。 常常看到一句话:在InnoDB中,除单个SQL组成的事务外,锁是逐步获得的。那就说明,单个SQL组成的事务锁是一次获得的。而此案例中,语句(2) 已经得到了主键索引的排他锁,为什么还会申请主键索引的排他锁呢?同理,语句(1)已经获得了主键索引的共享锁,为什么还会申请主键索引的共享锁呢? 死锁记录中,事务一等待锁的page no与事务二持有锁的page no相同,均为218436,这又代表什么呢? 我们的猜想是,innodb存储引擎中获得行锁是逐行获得的,并不是一次获得的。下面来证明。 死锁产生过程分析 要想知道innodb加锁的过程,唯一的方式就是运行mysql的debug版本,从gdb的输出中找到结果。根据gdb的结果得到,单个SQL组成的事 务,从宏观上来看,锁是在这个语句上一次获得的,但从底层实现上来看,是逐个记录行查询,得到符合条件的记录即对该行记录的索引加锁。 Gdb结果演示如下: 复制代码 代码如下: (gdb) b lock_rec_lock Breakpoint 1 at 0×867120: file lock/lock0lock.c,line 2070. (gdb) c Continuing. [Switching to Thread 1168550240 (LWP 5540)] Breakpoint 1,lock_rec_lock (impl=0,mode=5,rec=0x2aedbe01c1 “789200″,index=0x2aada734b8,thr=0x2aada74c18) at lock/lock0lock.c:2070 2070 { Current language: auto; currently c (gdb) c Continuing. Breakpoint 1,mode=1029,rec=0x2aedbc80ba “200″,index=0x2aada730b8,thr=0x2aada74c18) at lock/lock0lock.c:2070 2070 { (gdb) c Continuing. Breakpoint 1,rec=0x2aedbe01cf “789200″,thr=0x2aada74c18) at lock/lock0lock.c:2070 2070 { (gdb) c Continuing. (说明:”789200″为非聚簇索引,”200″为主键索引) Gdb结果显示,语句(1)(2)加锁的获取记录为多行,即逐行获得锁,这样就解释了语句(2)获得了主键索引锁还再次申请主键索引锁的情况。 于是,两个事务分别拥有部分锁并等待被对方持有的锁,出现这种资源循环等待的情况,即死锁。此案例中被检测时候的锁冲突就发现在page no为218436和218103的锁上。 当使用非聚簇索引时,会根据得到的主键值遍历聚簇索引,得到相应的记录。 条件: 主键索引锁冲突 主键索引锁与非聚簇索引锁冲突 此案例涉及TSK_TASK表,该表相关字段及索引如下: 条件: 当执行update、delete操作时,会修改表中的数据信息。由于innodb存储引擎中索引的数据存储结构,会根据修改语句使用的索引以及修改信息 的不同执行不同的加锁顺序。当使用索引进行查找并修改记录时,会首先加使用的索引锁,然后,如果修改了主键信息,会加主键索引锁和所有非聚簇索引锁,修改 了非聚簇索引列值会加该种非聚簇索引锁。 条件: 避免死锁的方法 (编辑:李大同) 【声明】本站内容均来自网络,其相关言论仅代表作者个人观点,不代表本站立场。若无意侵犯到您的权利,请及时与联系站长删除相关内容! |