1. Shared and Exclusive Lockshtml
shared lock (译:共享锁)mysql
exclusive lock (译:排它锁、独占锁)sql
InnoDB实现了标准的行级锁,其中有两种类型的锁,共享锁(shared locks)和排他锁(exclusive locks)。并发
A shared (S) lock permits the transaction that holds the lock to read a row.性能
An exclusive (X) lock permits the transaction that holds the lock to update or delete a row.spa
共享锁容许持有该锁的事务读取一行。code
排它锁容许持有该锁的事务更新或删除行。orm
若是事务T1持有行 r 上的共享锁(S),那么来自不一样事务T2的请求将按照如下方式处理: htm
若是事务T1持有行 r 上的排它锁(X),那么来自不一样事务T2的请求不能当即被授予 r 上任何一种类型的锁。相反,事务T2必须等待事务T1释放其在行 r 上的锁。blog
2. Intention Locks
Intention Locks(译:意向锁)
InnoDB支持多粒度锁,容许行锁和表锁共存。 例如,诸如LOCK TABLES ... WRITE之类的语句对指定表采用排它锁(X锁)。为了在多个粒度级别上实现锁,InnoDB使用了意向锁。意向锁是表级锁,它指示事务稍后须要对表中的一行使用哪一种类型的锁(共享锁或者排它锁)。
有两种类型的意向锁:
例如,SELECT ... LOCK IN SHARE MODE 设置一个IS锁,SELECT ... FOR UPDATE 设置一个IX锁。
意向锁的协定是这样的:
表级锁类型兼容性以下图:
若是一个锁与现有锁兼容,则将其授予请求的事务,但若是与现有锁冲突,则不授予该锁。事务等待,直到冲突的现有锁被释放。若是一个锁请求与一个现有的锁冲突,而且由于它会致使死锁而不能被授予,那么就会发生错误。
意向锁除了全表请求(例如LOCK TABLES ... WRITE)外,不阻止任何其余内容。意图锁定的主要目的是代表某人正在锁定表中的行或要锁定表中的行。
3. Record Locks
Record Locks(译:记录锁)
A record lock is a lock on an index record.
记录锁是索引记录上的锁。例如,SELECT c1 FROM t WHERE c1 = 10 FOR UPDATE; 防止任何其余事务插入、更新或删除t.c1值为10的行。
记录锁老是锁定索引记录,即便一个表没有定义索引也是如此。若是表没有索引,InnoDB建立一个隐藏的汇集索引,并将该索引用于记录锁。
4. Gap Locks
Gap Locks(译:间隙锁)
A gap lock is a lock on a gap between index records, or a lock on the gap before the first or after the last index record.
间隙锁是在索引记录之间的间隙上的锁,或者是在第一个索引记录以前或最后一个索引记录以后的间隙上的锁。
例如,SELECT c1 FROM t WHERE c1 BETWEEN 10 and 20 FOR UPDATE; 防止其余事务将值15插入到t.c1列中,不管该列中是否已经有这样的值,由于范围内全部现有值之间的间隙都被锁定了。
间隙可能跨越单个索引值、多个索引值,甚至是空的。
间隙锁是性能和并发性之间权衡的一部分,在某些事务隔离级别中使用,而在其余级别中则不使用。
对于使用惟一索引锁定行以搜索惟一行的语句,不须要间隙锁定。
例如,若是id列有一个惟一的索引,下面的语句只对id值为100的行使用index-record锁,而无论其余会话是否在前面的间隙插入行:
SELECT * FROM child WHERE id = 100;
若是id列没有索引或者有一个非惟一索引,则该语句会锁定前面的间隙。
这里还值得注意的是,不一样的事务能够在一个间隙上持有冲突的锁。
例如,事务A能够在一个间隙上持有一个共享间隙锁(gap S-lock),而事务B在同一个间隙上持有一个排他间隙锁(gap X-lock)。容许冲突间隙锁的缘由是,若是一条记录从一个索引中被清除,那么记录上由不一样事务持有的间隙锁必须被合并。
InnoDB中间隙锁的惟一目的是防止其余事务插入间隙。间隙锁能够共存。一个事务取得的间隙锁并不会阻止另外一个事务取得同一间隙上的间隙锁。共享和独占间隔锁之间没有区别。它们彼此之间不冲突,而且执行相同的功能。
5. Next-Key Locks
A next-key lock is a combination of a record lock on the index record and a gap lock on the gap before the index record.
next-key锁是索引记录上的记录锁和索引记录以前的间隙锁的组合。
InnoDB执行行级锁的方式是这样的:当它搜索或扫描一个表索引时,它会在遇到的索引记录上设置共享锁或排他锁。所以,行级锁其实是索引记录锁。索引记录上的next-key锁也会影响该索引记录以前的“间隙”。也就是说,next-key锁是索引记录锁加上索引记录以前的间隙锁。若是一个会话在一个索引中的记录R上有一个共享锁或排他锁,则另外一会话没法按照索引顺序在R以前的间隙中插入新的索引记录。
假设一个索引包含值十、十一、13和20。该索引可能的next-key锁覆盖如下区间:
(negative infinity, 10] (10, 11] (11, 13] (13, 20] (20, positive infinity)
默认状况下,InnoDB使用REPEATABLE READ事务隔离级别。在这种状况下,InnoDB使用next-key锁进行搜索和索引扫描,以阻止幻象行。
6. Insert Intention Locks
Insert Intention Locks(译:插入意向锁)
插入意向锁是一种间隙锁,是由INSERT操做在行插入以前设置的。这个锁表示,若是多个事务插入到同一个索引间隙中,若是它们没有插入到这个间隙中的同一位置,那么它们就不须要等待对方。假设有值为4和7的索引记录。尝试插入值分别为5和6的独立事务,在得到插入行的排他锁以前,每一个事务都用插入意向锁锁住4和7之间的间隙,但不会阻塞彼此,由于行是不冲突的。
7. AUTO-INC Locks
AUTO-INC锁是一种特殊的表级锁,由插入到带有AUTO_INCREMENT列的表中的事务得到。在最简单的状况下,若是一个事务正在向表中插入值,那么任何其余事务都必须等待本身对该表的插入,以便由第一个事务插入的行接收连续的主键值。