MySQL引擎之InnoDB

阅读更多

1、锁模型

  InnoDB有两种锁模型:共享锁、排它锁。

  共享锁(S):阻止其他链接行写,允许读取行,读取之前获得锁

  排他锁(X):阻止其他链接读取行,根据事物隔离级别决定是否可写入行,写入前获得锁

  事务在获取锁之前,需要获取意向锁(intention lock-I),意向锁是事务获取锁之前需要等待,目的是防止不同连接获得的锁在相同行上。

  锁之间的阻塞关系

  锁   被阻塞

  X     X, S, IX, IS

  S     X, IS

  IX    X, S

  IS    X

2、锁类型

表级锁

    锁住整张表,通过LOCK TABLES来获得表级锁,在实践中不提倡使用,在InnoDB中一般不使用。使用innodb_table_locks服务参数打开该功能,LOCK TABLES的目的是保证非事物表的一致性,比如Aria 和MyISAM。表级锁不仅仅是锁住事务中的行,而是锁住整个表,因此性能极差。

 

行级锁

   Record Lock: 作用于单个索引记录。如果表没有明确创建索引,锁将作用于聚集索引(clustered index)中的记录。

  Next-Key Lock:作用于一个索引记录及它之前的所有记录,这样可以防止其他连接插入或修改当前事务已访问的记录,只能用于重复读(REPEATABLE READ)

  Gap Lock:锁定一个结果集,集合可以是一条记录、多条记录及空集合。该锁不能用于使用UNIQUE索引查询的单条记录上,除非是多列索引。一个特殊示例是插入间隙锁,该锁在插入新行前获得,当其他连接试图使用相同索引值插入行时,它将挂起等待当前事务提交或者回滚。该锁不能用于读提交(READ COMMITTED)

锁诊断

  使用 SHOW ENGINE INNODB STATUS快速查看锁信息

  使用SHOW PROCESSLIST查看数据库进程信息

  锁信息存放位置information_schema.INNODB_LOCKS表中

  表INNODB_LOCK_WAITS中包含等待获取锁的事务,阻塞事务信息

3、读一致

不可重复读(The non-repeatable reads)

  在当前事务没有修改数据的情况下,执行两次查询得到不同的结果,引发原因是当前事务没有同其他修改请求的连接相隔离,侧重数据的修改。

 

幻读(Phantom reads)

  执行两次范围查询语句,得到的数据行记录不同,主要是数据的新增、删除影响。

 

一致性读(Consistent reads)

  不使用锁,当前线程第一次读表是创建一个快照,使用START TRANSACTION WITH CONSISTENT SNAPSHOT开启,当提交事务时,快照中的更改更新到表中,从而对所有连接可见。在重复读隔离级别中使用。

 

锁定读(Locking reads)

  比一致性读更强,需要锁表。在select语句中使用 LOCK IN SHARE MODE 或 FOR UPDATE。

  LOCK IN SHARE MODE阻止其他连接修改数据,但可以读。如果事务隔离级别为SERIALIZABLE且自动提交为FALSE,LOCK IN SHARE MODE总会被添加到select语句中。

  FOR UPDATE 排他锁(X)。

4、事务隔离级别

读未提交(READ UNCOMMITTED)

  在事务执行时为每个读命令创建一个单独的快照,读取使用数据快照,此事务中的数据可能包含未提交的数据,因此表中可能并不包含此数据

 

读提交(READ COMMITTED)

 在事务中为每个一致性读设置一个不同的快照,不会使用为提交的数据创建快照。更新、删除语句及锁定读不会使用间隙锁(gap lock)

 

重复读取(REPEATABLE READ)

  事务内的所有读使用相同快照,有比读提交更强的一致性约束,更新、删除语句及锁定读及record lock 使用唯一索引,gap lock及next-key lock 则会阻塞

 

串行化(SERIALIZABLE)

 可以认为重复读取模式下所有非锁定select自动转换为LOCK IN SHARE MODE,这两者是等价的:

SET TRANSACTION ISOLATION LEVEL REPEATABLE READ;
START TRANSACTION;
SELECT ....LOCK IN SHARE MODE;
UPDATE ...;
COMMIT;

SET TRANSACTION ISOLATION LEVEL SERIALIZABLE;
START TRANSACTION;
SELECT ...;
UPDATE ...;
COMMIT;

 

你可能感兴趣的:(mysql,引擎,InnoDB)