This article introduces the types of locks used by InnoDB, including global locks and table-level locks provided by mysql. InnoDB Locking is translated from the section of InnoDB Locking on the official website, adding additional global locks and table-level locks provided by mysql. The main purpose of this blog is to memorize and refresh knowledge. In the future, we will take a look at the transactions, which locks will be added to the transactions at different isolation levels, and we will sort them out in the form of experiments. At present, I haven’t written articles for a long time, and I am almost decadent. I insist on writing two articles every week.
Shared locks and exclusive locks
InnoDB implements standard row-level locks with two types of locks: shared (S) locks and exclusive (X) locks.
- A shared (S) lock allows the transaction holding the lock to read a row.
- An exclusive (X) lock allows the transaction holding the lock to update or delete a row.
If transaction T1 holds the shared (S) lock on row R, then a request from a different transaction T2 treats the lock on row R as follows:
- Transaction T2’s request to acquire the row R shared (S) lock can be approved immediately. As a result, transaction T1 and T2 both hold a shared (S) lock on row R
- Transaction T2’s request to acquire the row R exclusive (X) lock could not be approved immediately.
If transaction T1 holds an exclusive (X) lock on row R, a request from a different transaction T2 requesting either type of lock on row R cannot be immediately approved. Instead, transaction T2 must wait for transaction T1 to release the lock on row R. Mysql > select * from Mysql5.7; mysql > select * from Mysql5.7;
CREATE TABLE yangzai ( `id` int(11) NOT NULL AUTO_INCREMENT, `a` int(255) NULL DEFAULT NULL, `b` int(255) NULL DEFAULT NULL, `c` int(255) NULL DEFAULT NULL, PRIMARY KEY (`id`) USING BTREE, UNIQUE INDEX `un_a`(`a`) USING BTREE ) ENGINE = InnoDB AUTO_INCREMENT = 20 CHARACTER SET = utf8 INSERT into Yangzai (a, b, c) VALUES (1, 2, 3);Copy the code
The order of statements is the order in which SQL is executed
Session A starts the transaction and performs the INSERT operation. When inserting this data, session A will first use index A to check whether there is A unique index conflict. Yangzai table has been pre-initialized (1,2,3), session A fails to INSERT data, but session A still holds the shared lock of index A (two stages of lock, Before the update, the executor will call InnoDB engine interface to get the data of a=1, and add an exclusive lock on index A =1. Session B will wait for the exclusive lock on index A =1, because the shared lock of A =1 is held by session A and has not been released
Session began to perform A update statement, it is necessary to add an exclusive lock at A = 1 index of A session in A = 1 index with an exclusive lock waiting, because session B also to be in A = 1 index and an exclusive lock, session B waiting session in A = 1 Shared lock release of the index, the transaction and transaction B in the release, wait for each other’s resources is to enter the deadlock state.
When a deadlock occurs, there are two strategies: one is to simply wait until time out. This timeout can be set with the innodb_lock_WAIT_timeout parameter. Another strategy is to initiate deadlock detection, where a deadlock is found and one of the transactions in the chain is actively rolled back to allow the other transactions to continue. Setting innodb_deadlock_detect to ON indicates that this logic is enabled.
Second, intent lock
InnoDB supports multi-granularity locking, allowing row and table locking to coexist. For example, LOCK TABLES… Statements such as WRITE take an exclusive lock (X lock) on the specified table. To make multi-granularity locking practical, InnoDB uses intent locks. Intent locks are table-level locks that indicate what type of lock (shared or exclusive) the transaction later needs to use on the rows in the table. There are two types of intent locks:
- Intent shared lock (IS) indicates that the transaction intends to set a shared lock on a single row in the table.
- Intended exclusive lock (IX) indicates that the transaction intends to set an exclusive lock on a single row in the table.
For example, SELECT… FOR SHARE set an IS lock, SELECT… FOR UPDATE sets an IX lock. The intent lock protocol is as follows:
- Before a transaction can acquire a shared lock on a row in a table, it must first acquire an IS lock or a stronger IS lock on the table.
- Before a transaction can acquire an exclusive lock on a row in a table, it must first acquire the IX lock on that table.
The following table summarizes table-level lock type compatibility, that is, table lock compatibility with table-level intent locks.
If the requested transaction is compatible with the locks of an existing transaction, the lock is granted to that transaction, but if it conflicts with an existing lock, the transaction lock is not granted. The transaction will wait until the conflicting existing lock is released. An error occurs if the lock request conflicts with an existing lock and cannot be granted because it would cause a deadlock.
Intent locks only block full table requests (such as LOCK TABLES… WRITE/READ, if you want to set a shared LOCK on a row, LOCK TABLES… The WRITE operation will block. The intent lock is to allow both row locks and table locks to coexist. One transaction has already locked the row of the table exclusively, preventing other transactions from locking the table exclusively. The main purpose of an intent lock is to indicate that a transaction is locking a row or is about to lock a row in a table.
The transaction data for the intent lock is similar to the output of SHOW ENGINE INNODB STATUS or INNODB monitor below
TABLE LOCK table `test`.`t` trx id 10080 lock mode IXCopy the code
Record lock
A record lock is a lock on an index record. FOR example, SELECT c1 FROM t WHERE c1 = 10 FOR UPDATE; Prevents other transactions from inserting, updating, or deleting rows with the value 10 of T.c1. Record locks always lock index records, even if there is no index when the table is defined. In this case, InnoDB creates a hidden clustered index (primary key index) and uses this index to lock records. The transaction data for logging locks is similar to the output of SHOW ENGINE INNODB STATUS or INNODB monitor:
RECORD LOCKS space id 58 page no 3 n bits 72 index `PRIMARY` of table `test`.`t`
trx id 10078 lock_mode X locks rec but not gap
Record lock, heap no 2 PHYSICAL RECORD: n_fields 3; compact format; info bits 0
0: len 4; hex 8000000a; asc ;;
1: len 6; hex 00000000274f; asc 'O;; 2: len 7; hex b60000019d0110; asc ;;Copy the code
Four, clearance lock
A gap lock is a lock on the gap between index records, or before or after the first or last index record. FOR example, SELECT c1 FROM t WHERE c1 BETWEEN 10 and 20 FOR UPDATE; Prevents other transactions from inserting the value 15 into the T.c1 column, whether or not there are any such values in the column, because the gap between all existing values in the range is locked. Gaps can span a single index value, multiple index values, or even be empty. Gap locking is part of the trade-off between performance and concurrency and is used under some transaction isolation levels but not others. Gap locks are used in RR(repeatable read transaction isolation level, gap locks resolve phantom reads). A statement that locks a row using a unique index to search for a unique row does not require gap locking because equivalent searches using a unique index do not produce phantom reads. (This does not include search criteria that contain multiple unique indexes; In this case, gap locking occurs.) For example, if the ID column has a unique index, the following statement locks only the row with id 100 and does not affect whether other sessions are inserted:
SELECT * FROM child WHERE id = 100;Copy the code
If the ID has no index or the index is not unique, the statement locks the preceding gap. It is also worth noting here that different transactions may hold conflicting locks in a gap. For example, transaction A can hold A shared GAP lock (GAP S-lock) on another gap, while transaction B can hold an exclusive GAP lock (GAP X-lock) on the same gap. The reason for allowing conflicting gap locks is that if records are cleared from an index, then gap locks retained on records by different transactions must be merged. Gap locks in InnoDB are “completely forbidden”, meaning their sole purpose is to prevent other transactions from being inserted into the Gap. Gap locks can coexist. A gap lock performed by one transaction does not prevent another transaction from performing a gap lock on the same gap. There is no difference between shared and exclusive locks. They don’t conflict with each other, they perform the same function. Can display disable gap locking. This happens if you change the transaction isolation level to READ COMMITTED. In these cases, gap locking is disabled for searches and index scans, and is only used for foreign key constraint checking and duplicate key checking. Using the READ COMMITTED isolation level has other effects as well. After MySQL evaluates the WHERE condition, the record lock for the non-matching row is released. For UPDATE statements, InnoDB performs a “(semi-consistent) semi-consistent” read so that it returns the most recently committed version to MySQL so that MySQL can determine if the row matches the updated WHERE condition.
Five, the Next – Key Locks
A next-key lock is a combination of a record lock on an index record and a gap lock that precedes the index record. InnoDB performs row-level locking by setting shared or exclusive locks on index records it encounters when it searches or scans table indexes. Therefore, row-level locks are actually indexed record locks. A next-key lock on an index record also affects the gap in front of that index record. That is, a next-key lock is an index record lock plus a gap lock on the gap before the index record. If one session has a shared or exclusive lock on the record R in the index, another session cannot insert new index records in the gap before R in index order. Assume the index contains the values 10, 11, 13, and 20. The next-key of this index may override the following intervals, where a parenthesis indicates the exclusion of interval endpoints and a square bracket indicates the inclusion of endpoints:
(negative infinity, 10]
(10, 11]
(11, 13]
(13, 20]
(20, positive infinity)Copy the code
For the last interval, the next next-key lock locks the interval above the maximum value in the index, and the value of the “supremum” pseudo-record is higher than any actual value in the index. Upper bounds are not true index records, so, in effect, the next key lock locks only the gap after the maximum index value. By default, InnoDB runs at repeatable read transaction isolation level. In this case, InnoDB uses a next-key lock for search and index scans to prevent phantom rows. The transaction data for the next-key lock displays output similar to the following SHOW ENGINE INNODB STATUS or INNODB monitor:
RECORD LOCKS space id 58 page no 3 n bits 72 index `PRIMARY` of table `test`.`t`
trx id 10080 lock_mode X
Record lock, heap no 1 PHYSICAL RECORD: n_fields 1; compact format; info bits 0
0: len 8; hex 73757072656d756d; asc supremum;;
Record lock, heap no 2 PHYSICAL RECORD: n_fields 3; compact format; info bits 0
0: len 4; hex 8000000a; asc ;;
1: len 6; hex 00000000274f; asc 'O;; 2: len 7; hex b60000019d0110; asc ;;Copy the code
Insert intent lock
An insert intent lock is a gap lock set by an insert operation prior to row insertion. This lock represents the intent of the insert so that if multiple transactions inserted into the same index gap are not inserted at the same place in the gap, they do not have to wait for each other. Assume that there are index records with values 4 and 7. Try to insert separate transactions with values 5 and 6, respectively. Each transaction uses the insert intent lock to lock the gap between 4 and 7 before acquiring the exclusive lock for the inserted row, but does not block the other because the inserted rows do not conflict. The following example demonstrates obtaining the transaction that inserted the intent lock before acquiring the exclusive lock of the inserted record. This example involves two clients, A and B. Client A creates A table with two index records (90 and 102), and then starts A transaction that places an exclusive lock on index records with ids greater than 100. Exclusive locks include clearance locks prior to record 102:
mysql> CREATE TABLE child (id int(11) NOT NULL, PRIMARY KEY(id)) ENGINE=InnoDB;
mysql> INSERT INTO child (id) values (90),(102);
mysql> START TRANSACTION;
mysql> SELECT * FROM child WHERE id > 100 FOR UPDATE;
+-----+
| id |
+-----+
| 102 |
+-----+Copy the code
Client B starts a transaction, inserting a record into the gap. A transaction obtains an insert intent lock while waiting to acquire an exclusive lock.
mysql> START TRANSACTION;
mysql> INSERT INTO child (id) VALUES (101);Copy the code
The transaction data inserted with the intent lock shows output similar to the following SHOW ENGINE INNODB STATUS or INNODB monitor:
RECORD LOCKS space id 31 page no 3 n bits 72 index `PRIMARY` of table `test`.`child`
trx id 8731 lock_mode X locks gap before rec insert intention waiting
Record lock, heap no 3 PHYSICAL RECORD: n_fields 3; compact format; info bits 0
0: len 4; hex 80000066; asc f;;
1: len 6; hex 000000002215; asc "; 2: len 7; hex 9000000172011c; asc r ;; .Copy the code
Seven, self-increasing lock
The auto-Inc lock is a special table-level lock obtained by transaction inserts into a table with an AUTO_INCREMENT column. In the simplest case, if one transaction is inserting values into the table, any other transaction must wait for its own inserts into the table so that the row inserted by the first transaction receives consecutive primary key values. The innodb_autoINC_lock_mode configuration option controls the algorithm used for automatic incremental locking. It allows you to choose how to trade off predictable sequences of automatic increments against maximum concurrency for insert operations.
Predicate Locks for Spatial Indexes
InnoDB supports spatial indexing of columns containing spatial columns. To handle locking for operations involving spatial indexes, next-key locks do not well support transaction isolation levels of REPEATABLE READ or SERIALIZABLE. Multidimensional data has no concept of absolute ordering, so it is not clear which is the “next” key. To support isolation levels for tables with spatial indexes, InnoDB uses predicate locks. Spatial indexes contain minimum bound rectangle (MBR) values, so InnoDB enforces consistent reading of indexes by setting predicate locks on the MBR values used for queries. Other transactions cannot insert or modify rows that match the query criteria.
Global locking
A global lock is a lock on the entire database instance. MySQL provides a method for adding a global read lock by running Flush tables with read lock. When you use this command to make the entire library read-only, subsequent statements from other threads will block: data update statements (adding, deleting, or modifying data), data definition statements (including creating tables, modifying table structures, adding indexes, etc.), and commit statements for updating class transactions. The typical scenario of global locking is to perform a full database logical backup. Select text from each table in the database. The entire database is read-only during the backup process:
- If the backup is performed on the master database, the update operation cannot be performed during the backup, and the service can only be queried, basically stopping
- If the backup is performed on the secondary database, the binlog synchronized from the primary database cannot be executed on the secondary database during the backup, resulting in the master/slave delay. If most of the queries are performed on the secondary database, the queried data may be inconsistent.
The official logical backup tool is mysqldump. When mysqldump uses the — single-transaction argument, a transaction is started before the data is transferred to ensure that the consistency view is retrieved. Due to the support of MVCC, data can be updated normally during this process. Consistent reads require that the engine support this isolation level. MyISAM does not support transactions. If the backup process has updated data, only the latest data can be retrieved, which will destroy the consistency of the backup. You can only use global locks to ensure consistency of backups. Set global readonly=true can also be used to make the whole database read-only. Readonly can be used to make the whole database read-only, but the global lock is recommended for backup consistency:
- Readonly is used to determine whether a library is primary or secondary.
- Client exception handling mechanism. If the global lock client is abnormally disconnected, Mysql will automatically release the global lock, and the database will be updated normally. After the entire library is set to Readonly, the database will remain in readOnly state if a client exception occurs, causing the entire library to be unwritable for a long time.
Table level locking
Mysql table level locks have two types: table locks and metadata locks. The syntax for table locking is lock tables… Read/Write, similar to global locks, you can unlock tables to release the lock. MyISAM does not support row locking and only supports minimal table-level locking. In addition to limiting reads and writes by other threads, Lock tables also limits what the thread can manipulate next. For example, if you execute lock tables T1 read, T2 write on thread A; This statement blocks statements that write to or write to T1 or T2 from other threads. In addition, thread A can only read T1 and read T2 before executing unlock tables. It is not allowed to write to T1, so it is not allowed to access other tables. Metadata table level locks, which do not require display locks, are automatically locked when a table is accessed. Prevents DDL and DML from running simultaneously in different threads. If a query is traversing a table and another thread changes the table structure or deletes a column during execution, the query thread will get data that does not match the table structure. There are two types of metadata table level locks: metadata table level read locks and metadata table level write locks.
- Metadata table-level read locks are not mutually exclusive and allow multiple threads to add, delete, modify, or query the same table at the same time.
- Metadata table level read lock Write lock, write lock write lock mutually exclusive, to ensure that the table structure changes. If multiple threads are adding fields to the table at the same time, wait for the other threads to finish before starting to execute.
Use the show the processlist. Viewing Connection Information
Write wrong place or bad place hope everybody helps correct next