Dieser Artikel vermittelt Ihnen relevantes Wissen über MySQL, das hauptsächlich das relevante Wissen über Sperren auf Zeilenebene in InnoDB einführt. Wie der Name schon sagt, handelt es sich um Sperren, die Datensätzen hinzugefügt werden Wenn ich es zusammenstelle, hoffe ich, dass es für alle hilfreich sein wird.
Empfohlenes Lernen: MySQL-Video-Tutorial
Zeilensperre, auch Datensatzsperre genannt, ist, wie der Name schon sagt, eine dem Datensatz hinzugefügte Sperre. Bitte beachten Sie jedoch, dass sich dieser Datensatz auf das Sperren des Indexeintrags im Index bezieht. Diese Zeilensperren-Implementierungsfunktion von InnoDB bedeutet, dass InnoDB Sperren auf Zeilenebene nur verwendet, wenn Daten über Indexbedingungen abgerufen werden. Andernfalls verwendet InnoDB Tabellensperren.
Ganz gleich, ob Sie einen Primärschlüsselindex, einen eindeutigen Index oder einen normalen Index verwenden, InnoDB verwendet Zeilensperren, um Daten zu sperren.
Zeilensperren können nur verwendet werden, wenn der Ausführungsplan tatsächlich den Index verwendet: Auch wenn das Indexfeld in der Bedingung verwendet wird, bestimmt MySQL anhand der Kosten verschiedener Ausführungspläne, ob der Index zum Abrufen von Daten verwendet werden soll MySQL betrachtet das Scannen der gesamten Tabelle als effizienter. Bei einigen sehr kleinen Tabellen werden beispielsweise keine Indizes verwendet. In diesem Fall verwendet InnoDB Tabellensperren anstelle von Zeilensperren.
Wenn wir zum Abrufen von Daten Bereichsbedingungen anstelle von Gleichheitsbedingungen verwenden und eine Sperre anfordern, sperrt InnoDB gleichzeitig die Indexelemente vorhandener Datensätze, die die Bedingungen erfüllen.
Aber auch Zeilensperren werden in InnoDB in verschiedene Typen unterteilt. Mit anderen Worten: Selbst wenn eine Zeilensperre zum selben Datensatz hinzugefügt wird, ist der Effekt unterschiedlich, wenn der Typ unterschiedlich ist.
Hier verwenden wir immer noch die vorherige Lehrertabelle, fügen einen Index hinzu und fügen einige Datensätze ein.
mysql> desc teacher; +--------+--------------+------+-----+---------+-------+ | Field | Type | Null | Key | Default | Extra | +--------+--------------+------+-----+---------+-------+ | number | int(11) | NO | PRI | NULL | | | name | varchar(100) | YES | MUL | NULL | | | domain | varchar(100) | YES | | NULL | | +--------+--------------+------+-----+---------+-------+ 3 rows in set (0.00 sec) mysql> select * from teacher; +--------+------+--------+ | number | name | domain | +--------+------+--------+ | 1 | T | Java | | 3 | M | Redis | | 9 | X | MQ | | 15 | O | Python | | 21 | A | Golang | +--------+------+--------+ 5 rows in set (0.00 sec)
Werfen wir einen Blick auf die gängigen Arten von Zeilensperren.
Sie werden auch Datensatzsperren genannt und sperren nur einen Datensatz. Der offizielle Typname lautet: LOCK_REC_NOT_GAP. Das schematische Diagramm zum Hinzufügen einer Datensatzsperre zum Datensatz mit einem Zahlenwert von 9 sieht beispielsweise wie folgt aus:
Datensatzsperren werden in S-Sperren und X-Sperren unterteilt, wenn eine Transaktion eine S-Typ-Datensatzsperre erhält für einen Datensatz Danach können andere Transaktionen weiterhin die Datensatzsperre vom Typ S des Datensatzes erwerben, sie können jedoch nicht weiterhin die Datensatzsperre vom Typ X erwerben, wenn eine Transaktion die Datensatzsperre vom Typ X eines Datensatzes erhält Transaktionen können die X-Typ-Datensatzsperre des Datensatzes nicht mehr erhalten, und X-Typ-Datensatzsperren können nicht mehr erhalten werden.
wählen Sie * vom Lehrer wo number=9 für update; # Blocking | |
---|---|
Gap Locks | MySQL kann das Phantom-Read-Problem unter der Isolationsstufe REPEATABLE READ teilweise lösen. Es gibt zwei Lösungen. Bei der Verwendung der Sperrlösung besteht jedoch ein Problem: Wenn die Transaktion zum ersten Mal den Lesevorgang ausführt, sind diese Phantomdatensätze noch nicht vorhanden und wir können diesen Phantomdatensätzen keine Datensatzsperren hinzufügen. InnoDB schlägt einen Sperrtyp namens Gap Locks vor. Der offizielle Typname lautet: LOCK_GAP. Wir können ihn auch kurz Gap Lock nennen. | Gap Lock sperrt im Wesentlichen die Lücke vor und nach dem Index, sperrt jedoch nicht den Index selbst.
T1 | T2|
begin; |
update teacher set domain='Redis' where name='M';
einfügen into teacher value(23,'B','docker'); # Blocking | |
---|---|
insert into teacher value(23,'B','docker'); 事务T1会对([A, 21], [M, 3])、([M, 3], [O, 15])之间进行上gap锁,如下图中所示: 意味着不允许别的事务在这条记录前后间隙插入新记录,所以T2就不能插入。 但是当SQL语句变为: insert into teacher value(70,'P','docker'); Nach dem Login kopieren 能插入吗?当然能,因为(70,‘P’)这条记录不在被锁的区间内。 思考题现在有表,表中有记录如下: <span style="font-family: " microsoft yahei sans gb helvetica neue tahoma arial sans-serif>list = ['su liang','hacker','ice']<br>list.insert(1,'kiko')<br>print(list)<br>#结果:['su liang', 'kiko', 'hacker', 'ice']</span><br> Nach dem Login kopieren 开启一个事务: begin;SELECT * FROM test1 WHERE number = 3 FOR UPDATE; Nach dem Login kopieren 开启另外一个事务: INSERT INTO test1 (id, number) VALUES (2, 1); # 阻塞 INSERT INTO test1 (id, number) VALUES (3, 2); # 阻塞 INSERT INTO test1 (id, number) VALUES (6, 8); # 阻塞 INSERT INTO test1 (id, number) VALUES (8, 8); # 正常执行 INSERT INTO test1 (id, number) VALUES (9, 9); # 正常执行 INSERT INTO test1 (id, number) VALUES (10, 12); # 正常执行 UPDATE test1 SET number = 5 WHERE id = 11 AND number = 12; # 阻塞 Nach dem Login kopieren 为什么(6,8)不能执行,(8,8)可以执行?这个间隙锁的范围应该是[1,8],最后一个语句为什么不能执行?
Next-Key Locks有时候我们既想锁住某条记录,又想阻止其他事务在该记录前边的间隙插入新记录,所以InnoDB就提出了一种称之为Next-Key Locks的锁,官方的类型名称为:LOCK_ORDINARY,我们也可以简称为next-key锁。next-key锁的本质就是 默认情况下,InnoDB以REPEATABLE READ隔离级别运行。在这种情况下,InnoDB使用Next-Key Locks锁进行搜索和索引扫描,这可以防止幻读的发生。 Insert Intention Locks我们说一个事务在插入一条记录时需要判断一下插入位置是不是被别的事务加了所谓的gap锁(next-key锁也包含gap 锁,后边就不强调了),如果有的话,插入操作需要等待,直到拥有gap锁的那个事务提交。 但是InnoDB规定事务在等待的时候也需要在内存中生成一个锁结构,表明有事务想在某个间隙中插入新记录,但是现在处于等待状态。这种类型的锁命名为Insert Intention Locks,官方的类型名称为:LOCK_INSERT_INTENTION,我们也可以称为插入意向锁。 可以理解为插入意向锁是一种锁的的等待队列,让等锁的事务在内存中进行排队等待,当持有锁的事务完成后,处于等待状态的事务就可以获得锁继续事务了。 隐式锁锁的的维护是需要成本的,为了节约资源,MySQL在设计提出了了一个隐式锁的概念。一般情况下INSERT操作是不加锁的,当然真的有并发冲突的情况下下,还是会导致问题的。 所以MySQL中,一个事务对新插入的记录可以不显式的加锁,但是别的事务在对这条记录加S锁或者X锁时,会去检查索引记录中的trx_id隐藏列,然后进行各种判断,会先帮助当前事务生成一个锁结构,然后自己再生成一个锁结构后进入等待状态。但是由于事务id的存在,相当于加了一个隐式锁。 这样的话,隐式锁就起到了延迟生成锁的用处。这个过程,我们无法干预,是由引擎自动处理的,对我们是完全透明的,我们知道下就行了。 锁的内存结构所谓的锁其实是一个内存中的结构,在事务执行前本来是没有锁的,也就是说一开始是没有锁结构和记录进行关联的,当一个事务想对这条记录做改动时,首先会看看内存中有没有与这条记录关联的锁结构,当没有的时候就会在内存中生成一个锁结构与之关联。比方说事务T1要对记录做改动,就需要生成一个锁结构与之关联。 锁结构里至少要有两个比较重要的属性:
当事务T1改动了条记录后,就生成了一个锁结构与该记录关联,因为之前没有别的事务为这条记录加锁,所以is_waiting 属性就是false,我们把这个场景就称之为获取锁成功,或者加锁成功,然后就可以继续执行操作了。 在事务T1提交之前,另一个事务T2也想对该记录做改动,那么先去看看有没有锁结构与这条记录关联,发现有一个锁结构与之关联后,然后也生成了一个锁结构与这条记录关联,不过锁结构的is_waiting属性值为true,表示当前事务需要等待,我们把这个场景就称之为获取锁失败,或者加锁失败,或者没有成功的获取到锁。 在事务T1提交之后,就会把该事务生成的锁结构释放掉,然后看看还有没有别的事务在等待获取锁,发现了事务T2还在等待获取锁,所以把事务T2对应的锁结构的is_waiting属性设置为false,然后把该事务对应的线程唤醒,让它继续执行,此时事务T2就算获取到锁了。这种实现方式非常像并发编程里AQS的等待队列。 对一条记录加锁的本质就是在内存中创建一个锁结构与之关联。那么,一个事务对多条记录加锁时,是不是就要创建多个锁结构呢?比如: SELECT * FROM teacher LOCK IN SHARE MODE; Nach dem Login kopieren 很显然,这条语句需要为teacher表中的所有记录进行加锁。那么,是不是需要为每条记录都生成一个锁结构呢?其实理论上创建多个锁结构没有问题,反而更容易理解。但是如果一个事务要获取10,000条记录的锁,要生成10,000个这样的结构,不管是执行效率还是空间效率来说都是很不划算的,所以实际上,并不是一个记录一个锁结构。 当然锁结构实际是很复杂的,我们大概了解下里面包含哪些元素。
基本上来说,同一个事务里,同一个数据页面,同一个加锁类型的锁会保存在一起。 推荐学习:mysql视频教程 |
Das obige ist der detaillierte Inhalt vonMySQL-Wissenspunkte: Sperren auf Zeilenebene in InnoDB. Für weitere Informationen folgen Sie bitte anderen verwandten Artikeln auf der PHP chinesischen Website!