现在的位置: 首页 > 数据库 > 正文

MySQL死锁问题原因有哪些?MySQL死锁问题怎么解决

2020年04月28日 数据库 ⁄ 共 1602字 ⁄ 字号 评论关闭

  MySQL死锁问题是每个IT人员都会面临的问题,有的人员技术老练很快就能解决,但对于新手来说这个问题就比较困难了。下面学步园小编来讲解下MySQL死锁问题原因有哪些?MySQL死锁问题怎么解决?

  MySQL死锁问题原因有哪些

  1、MySQL常用存储引擎的锁机制

  MyISAM和MEMORY采用表级锁(table-levellocking)

  BDB采用页面锁(page-levellocking)或表级锁,默认为页面锁

  InnoDB支持行级锁(row-levellocking)和表级锁,默认为行级锁

  2、各种锁特点

  表级锁:开销小,加锁快;不会出现死锁;锁定粒度大,发生锁冲突的概率最高,并发度最低

  行级锁:开销大,加锁慢;会出现死锁;锁定粒度最小,发生锁冲突的概率最低,并发度也最高

  页面锁:开销和加锁时间界于表锁和行锁之间;会出现死锁;锁定粒度界于表锁和行锁之间,并发度一般

  3、各种锁的适用场景

  表级锁更适合于以查询为主,只有少量按索引条件更新数据的应用,如Web应用

  行级锁则更适合于有大量按索引条件并发更新数据,同时又有并发查询的应用,如一些在线事务处理系统

  4、死锁

  是指两个或两个以上的进程在执行过程中,因争夺资源而造成的一种互相等待的现象,若无外力作用,它们都将无法推进下去。

  表级锁不会产生死锁.所以解决死锁主要还是针对于最常用的InnoDB.

  5、死锁举例分析

  在MySQL中,行级锁并不是直接锁记录,而是锁索引。索引分为主键索引和非主键索引两种,如果一条sql语句操作了主键索引,MySQL就会锁定这条主键索引;如果一条语句操作了非主键索引,MySQL会先锁定该非主键索引,再锁定相关的主键索引。

  在UPDATE、DELETE操作时,MySQL不仅锁定WHERE条件扫描过的所有索引记录,而且会锁定相邻的键值,即所谓的next-keylocking。

  例如,一个表db.tab_test,结构如下:

  id:主键;

  state:状态;

  time:时间;

  索引:idx_1(state,time)

  出现死锁日志如下:

  ***(1)TRANSACTION:TRANSACTION0677833455,ACTIVE0sec,processno11393,OSthreadid278546startingindexreadmysqltablesinuse1,locked1LOCKWAIT3lockstruct(s),heapsize320MySQLthreadid83,queryid162348740dcnet03dcnetSearchingrowsforupdateupdatetab_testsetstate=1064,time=now()wherestate=1061andtime

  原因分析:

  当“updatetab_testsetstate=1064,time=now()wherestate=1061andtime

  假设“updatetab_testsetstate=1067,time=now()whereidin(9921180)”几乎同时执行时,本语句首先锁定主键索引,由于需要更新state的值,所以还需要锁定idx_1的某些索引记录。

  这样第一条语句锁定了idx_1的记录,等待主键索引,而第二条语句则锁定了主键索引记录,而等待idx_1的记录,这样死锁就产生了。

  MySQL死锁问题怎么解决

  拆分第一条sql,先查出符合条件的主键值,再按照主键更新记录:

  selectidfromtab_testwherestate=1061andtime

  以上就是关于“MySQL死锁问题原因有哪些?MySQL死锁问题怎么解决”的内容,希望对大家有用。更多资讯请关注学步园。学步园,您学习IT技术的优质平台!

抱歉!评论已关闭.