A brief discussion on the manifestation and value selection method of innodb_autoinc_lock_mode

A brief discussion on the manifestation and value selection method of innodb_autoinc_lock_mode

Prerequisite: Percona 5.6 version, transaction isolation level is RR

mysql> show create table test_autoinc_lock\G
*************************** 1. row ***************************
    Table: test_autoinc_lock
Create Table: CREATE TABLE `test_autoinc_lock` (
 `id` int(11) NOT NULL AUTO_INCREMENT,
 `a` int(11) DEFAULT NULL,
 PRIMARY KEY (`id`),
 KEY `idx_a` (`a`)
) ENGINE=InnoDB AUTO_INCREMENT=14 DEFAULT CHARSET=utf8

1 row in set (0.00 sec)
mysql> select * from test_autoinc_lock;
+----+------+
| id | a |
+----+------+
| 1 | 1 |
| 12 | 2 |
| 2 | 3 |
| 3 | 5 |
| 4 | 7 |
| 5 | 7 |
| 6 | 9 |
| 7 | 10 |
+----+------+
8 rows in set (0.00 sec)

Condition 1 innodb_autoinc_lock_mode is set to 0

session1
 begin;delete from test_autoinc_lock where a>7;//session2 is not submitted at this time
mysql> insert into test_autoinc_lock(a) values(100); //Gap lock exists, and the lock is waiting for session 3
mysql> insert into test_autoinc_lock(a) values(2); //This is also in the waiting state. In theory, this is not the locking range of the gap lock, so what is it waiting for? session4
mysql> select * from information_schema.innodb_trx\G
*************************** 1. row ***************************
          trx_id: 2317
         trx_state: LOCK WAIT
        trx_started: 2016-10-31 19:28:05
   trx_requested_lock_id: 2317:20
     trx_wait_started: 2016-10-31 19:28:05
        trx_weight: 1
    trx_mysql_thread_id: 9
         trx_query: insert into test_autoinc_lock(a) values(2)
    trx_operation_state: setting auto-inc lock
     trx_tables_in_use: 1
     trx_tables_locked: 1
     trx_lock_structs: 1
   trx_lock_memory_bytes: 360
      trx_rows_locked: 0
     trx_rows_modified: 0
  trx_concurrency_tickets: 0
    trx_isolation_level: REPEATABLE READ
     trx_unique_checks: 1
  trx_foreign_key_checks: 1
trx_last_foreign_key_error: NULL
 trx_adaptive_hash_latched: 0
 trx_adaptive_hash_timeout: 10000
     trx_is_read_only: 0
trx_autocommit_non_locking: 0

At this time, check that session 3 is waiting for the auto-increment lock and has been in the setting auto-inc lock state.

session2

ERROR 1205 (HY000): Lock wait timeout exceeded; try restarting transaction

At this time, session3 locks and waits for timeout to exit

session3

Now look at session3 and you can see that the insert is complete.

mysql> select * from test_autoinc_lock;
+----+------+
| id | a |
+----+------+
| 1 | 1 |
| 12 | 2 |
| 13 | 2 |
| 2 | 3 |
| 3 | 5 |
| 4 | 7 |
| 5 | 7 |
| 6 | 9 |
| 7 | 10 |
+----+------+
9 rows in set (0.00 sec) //Note that the maximum auto-increment value at this time is 13, which is the previous maximum auto-increment value + 1. In other words, session2 later released the expected auto-increment id and left 13 to session3. The application of auto-increment id values ​​is completely serial.

Conclusion: When innodb_autoinc_lock_mode is 0, it is officially called traditional

Level, the auto-increment lock is at the table lock level, and it must wait until the current SQL is executed or rolled back before it is released. In this way, it is conceivable that the competition for the auto-increment lock is relatively large in the case of high concurrency.

Condition 2 innodb_autoinc_lock_mode is set to 1

session1
mysql> begin;
Query OK, 0 rows affected (0.00 sec)


mysql> delete from test_autoinc_lock where a>7;
Query OK, 2 rows affected (0.00 sec)
mysql> select * from test_autoinc_lock;
+----+------+
| id | a |
+----+------+
| 1 | 1 |
| 12 | 2 |
| 13 | 2 |
| 2 | 3 |
| 3 | 5 |
| 4 | 7 |
| 5 | 7 |
| 6 | 9 |
| 7 | 10 |
+----+------+
9 rows in set (0.00 sec) //Note that the maximum auto-increment at this time is 13


session2
mysql> insert into test_autoinc_lock(a) values(100); //The same gap lock exists, and the lock is waiting for session 3
mysql> insert into test_autoinc_lock(a) values(5);
Query OK, 1 row affected (0.00 sec)


mysql> select * from test_autoinc_lock;
+----+------+
| id | a |
+----+------+
| 1 | 1 |
| 12 | 2 |
| 13 | 2 |
| 2 | 3 |
| 3 | 5 |
| 15 | 5 |
| 4 | 7 |
| 5 | 7 |
| 6 | 9 |
| 7 | 10 |
+----+------+
10 rows in set (0.00 sec) //session3 is completed directly, and note that the inserted auto-increment id value is 15, which means that the 14 expected to be assigned to session2 is skipped. You can see that the auto-increment id value is immediately assigned to session3 without having to wait for session2 to complete execution.

Conclusion: When innodb_autoinc_lock_mode is 1, it is officially called consecutive

At this level, if it is a single insert SQL, the lock can be acquired and released immediately without waiting for the current SQL to be executed (unless a session has already acquired the auto-increment lock in other transactions). In addition, when the SQL is some batch insert SQL, such as insert into ...select ..., load data, replace ..select.., it is still a table-level lock, which can be understood as degenerating into having to wait for the current SQL to be executed before being released.

It can be considered that when the value is 1, it is a relatively light lock and will not affect replication. The only drawback is that the generated auto-increment value may not be completely continuous (but I personally think this is often not very important, and there is no need to count the number of rows based on the auto-increment id value).

Condition 3: innodb_autoinc_lock_mode is set to 2

Let me first state the conclusion: when innodb_autoinc_lock_mode is set to 2, all insert-type SQL statements can immediately acquire and release locks, which is the most efficient. However, a new problem will be introduced: when binlog_format is statement, the replication cannot be guaranteed to be safe, because batch insert, such as insert ..select... statements, can also immediately obtain a large number of auto-increment id values ​​in this case without locking the entire table. The slave will inevitably be confused when replaying this SQL. Let's do a test to verify that replication is not safe.

master session1
mysql> show variables like '%binlog_for%';
+---------------+-----------+
| Variable_name | Value |
+---------------+-----------+
| binlog_format | STATEMENT |
+---------------+-----------+
1 row in set (0.00 sec)
mysql> insert into test_autoinc_lock(a) select * from test_auto;
Query OK, 8388608 rows affected, 1 warning (29.85 sec)
Records: 8388608 Duplicates: 0 Warnings: 1


master session2 (note that session2 is executed before session1 is completed)
mysql> insert into test_autoinc_lock(a) values(2);
Query OK, 1 row affected (0.01 sec)
mysql> select * from test_autoinc_lock where a=2;
+---------+------+
| id | a |
+---------+------+
| 1376236 | 2 |
+---------+------+
1 row in set (0.00 sec)


slave session1 (you can see 1376236 primary key conflicts at this time)
mysql> show slave status\G
*************************** 1. row ***************************
        Slave_IO_State: Waiting for master to send event
         Master_Host: 10.9.73.139
         Master_User: ucloudbackup
         Master_Port: 3306
        Connect_Retry: 60
       Master_Log_File: mysql-bin.000006
     Read_Master_Log_Pos: 75823243
        Relay_Log_File:mysql-relay.000002
        Relay_Log_Pos: 541
    Relay_Master_Log_File: mysql-bin.000006
       Slave_IO_Running: Yes
      Slave_SQL_Running: No
       Replicate_Do_DB: 
     Replicate_Ignore_DB: 
      Replicate_Do_Table: 
    Replicate_Ignore_Table: 
   Replicate_Wild_Do_Table: 
 Replicate_Wild_Ignore_Table: 
          Last_Errno: 1062
          Last_Error: Error 'Duplicate entry '1376236' for key 'PRIMARY'' on query. Default database: 'test'. Query: 'insert into test_autoinc_lock(a) select * from test_auto'
         Skip_Counter: 0
     Exec_Master_Log_Pos: 75822971

We can easily find the cause of the problem by analyzing the binlog of the master database. When the first batch insert was not completed, the second simple insert obtained a lock with an auto-increment id value of 1376236. There is no problem writing to the master database at this time, but when it is reflected in the slave database, a primary key conflict is bound to occur because it is statement-based replication.

SET INSERT_ID=1376236/*!*/;
#161031 21:44:31 server id 168380811 end_log_pos 75822940 CRC32 0x65797f1c Query thread_id=20 exec_time=0 error_code=0
use `test`/*!*/;
SET TIMESTAMP=1477921471/*!*/;
insert into test_autoinc_lock(a) values(2)
/*!*/;
# at 75822940
#161031 21:44:31 server id 168380811 end_log_pos 75822971 CRC32 0xbb91449d Xid = 274
COMMIT /*!*/;
# at 75822971
#161031 21:44:26 server id 168380811 end_log_pos 75823050 CRC32 0xa297b57b Query thread_id=57 exec_time=30 error_code=0
SET TIMESTAMP=1477921466/*!*/;
BEGIN
/*!*/;
# at 75823050
# at 75823082
#161031 21:44:26 server id 168380811 end_log_pos 75823082 CRC32 0xa5aa31a1 Intvar
SET INSERT_ID=1/*!*/;
#161031 21:44:26 server id 168380811 end_log_pos 75823212 CRC32 0x470282ba Query thread_id=57 exec_time=30 error_code=0
SET TIMESTAMP=1477921466/*!*/;
insert into test_autoinc_lock(a) select * from test_auto

Summarize:

1 When copying innodb rows, you can set innodb_autoinc_lock_mode to 2, which will maximize the concurrency of the table in all insert situations.

2 When replicating innodb statements, you can set innodb_autoinc_lock_mode to 1 to ensure replication security while achieving maximum concurrency for simple insert statements.

3 In the case of the MyISAM engine, no matter what kind of auto-increment ID lock is a table-level lock, setting the innodb_autoinc_lock_mode parameter is invalid (test omitted)

4 In fact, the questioner mentioned that in the case of using the auto-increment ID value as the primary key under the InnoDB engine, the insertion speed can be improved compared to UUID or a custom primary key, because InnoDB is a primary key clustered index, and the actual primary key value must be accessed in the order of the primary key. Then the auto-increment ID itself is in ascending order, so when inserting data, the underlying layer does not need to perform additional sorting operations, and the number of index page splits is reduced, thereby greatly increasing the insert speed (unless other solutions can also ensure that the primary key is completely auto-incremented)

The above brief discussion on the manifestation and value selection reference method of innodb_autoinc_lock_mode is all the content that the editor shares with you. I hope it can give you a reference. I also hope that you will support 123WORDPRESS.COM.

<<:  Detailed explanation of several ways to write private variables of ES6 implementation class

>>:  Initial settings after installing Ubuntu 16 in the development environment

Recommend

Detailed explanation of Nginx process scheduling problem

Nginx uses a fixed number of multi-process models...

Analysis and Solution of ERROR:2002 Reported When MySQL Starts

Preface This article mainly introduces the analys...

Basic operation tutorial of files and permissions in centos

Preface Before we begin, we should briefly unders...

HTML n ways to achieve alternate color code sample code

This article mainly introduces the sample code of...

A brief analysis of MySQL parallel replication

01 The concept of parallel replication In the mas...

WeChat applet implements search function and jumps to search results page

Search Page: search.wxml page: <view class=&qu...

Introduction to deploying selenium crawler program under Linux system

Table of contents Preface 1. What is selenium? 2....

How to solve the Docker container startup failure

Question: After the computer restarts, the mysql ...

MySQL Failover Notes: Application-Aware Design Detailed Explanation

1. Introduction As we all know, in the applicatio...

Difference between querySelector and getElementById methods in JS

Table of contents 1. Overview 1.1 Usage of queryS...

Drawing fireworks effect of 2021 based on JS with source code download

This work uses the knowledge of front-end develop...

Detailed explanation of TIMESTAMPDIFF case in MySQL

1. Syntax TIMESTAMPDIFF(unit,begin,end); Returns ...

CSS3 realizes the red envelope shaking effect

There is a requirement to realize the shaking eff...