A complete record of a Mysql deadlock troubleshooting process

A complete record of a Mysql deadlock troubleshooting process

Preface

The database deadlocks I encountered before were all caused by inconsistent locking order during batch updates, but last week I encountered a deadlock that was very difficult to understand. I took this opportunity to relearn MySQL deadlock knowledge and common deadlock scenarios. After much research and discussions with colleagues, I finally discovered the cause of the deadlock problem and gained a lot. Although we are backend programmers, we do not need to analyze the lock-related source code as deeply as DBAs, but if we can master the basic deadlock troubleshooting methods, it will be of great benefit to our daily development.

PS: This article will not introduce the basic knowledge of deadlock. For the locking principle of MySQL, please refer to the link provided in the reference materials of this article.

Deadlock Causes

Let me first introduce the database and table situation. Because it involves real data within the company, the following is simulated and will not affect the specific analysis.

We use the MySQL database version 5.5, the transaction isolation level is the default RR (Repeatable-Read), and the InnoDB engine is used. Assume that there is a test table:

CREATE TABLE `test` (
 `id` int(11) unsigned NOT NULL AUTO_INCREMENT,
 `a` int(11) unsigned DEFAULT NULL,
 PRIMARY KEY (`id`),
 UNIQUE KEY `a` (`a`)
) ENGINE=InnoDB AUTO_INCREMENT=100 DEFAULT CHARSET=utf8;

The structure of the table is very simple, with a primary key id and another unique index a. The data in the table are as follows:

mysql> select * from test;
+----+------+
| id | a |
+----+------+
| 1 | 1 |
| 2 | 2 |
| 4 | 4 |
+----+------+
3 rows in set (0.00 sec)

The operations that cause deadlock are as follows:

step Transaction 1 Transaction 2
1 begin
2 delete from test where a = 2;
3 begin
4 delete from test where a = 2; (Transaction 1 is stuck)
5 Deadlock occurs: ERROR 1213 (40001): Deadlock found when trying to get lock; try restarting transaction insert into test (id, a) values ​​(10, 2);

Then we can view the deadlock log through SHOW ENGINE INNODB STATUS; ;:

------------------------
LATEST DETECTED DEADLOCK
------------------------
170219 13:31:31
*** (1) TRANSACTION:
TRANSACTION 2A8BD, ACTIVE 11 sec starting index read
mysql tables in use 1, locked 1
LOCK WAIT 2 lock struct(s), heap size 376, 1 row lock(s)
MySQL thread id 448218, OS thread handle 0x2abe5fb5d700, query id 18923238 renjun.fangcloud.net 121.41.41.92 root updating
delete from test where a = 2
*** (1) WAITING FOR THIS LOCK TO BE GRANTED:
RECORD LOCKS space id 0 page no 923 n bits 80 index `a` of table `oauthdemo`.`test` trx id 2A8BD lock_mode X waiting
Record lock, heap no 3 PHYSICAL RECORD: n_fields 2; compact format; info bits 32
 0: len 4; hex 00000002; asc ;;
 1: len 4; hex 00000002; asc ;;
*** (2) TRANSACTION:
TRANSACTION 2A8BC, ACTIVE 18 sec inserting
mysql tables in use 1, locked 1
4 lock struct(s), heap size 1248, 3 row lock(s), undo log entries 2
MySQL thread id 448217, OS thread handle 0x2abe5fd65700, query id 18923239 renjun.fangcloud.net 121.41.41.92 root update
insert into test (id,a) values ​​(10,2)
*** (2) HOLDS THE LOCK(S):
RECORD LOCKS space id 0 page no 923 n bits 80 index `a` of table `oauthdemo`.`test` trx id 2A8BC lock_mode X locks rec but not gap
Record lock, heap no 3 PHYSICAL RECORD: n_fields 2; compact format; info bits 32
 0: len 4; hex 00000002; asc ;;
 1: len 4; hex 00000002; asc ;;
*** (2) WAITING FOR THIS LOCK TO BE GRANTED:
RECORD LOCKS space id 0 page no 923 n bits 80 index `a` of table `oauthdemo`.`test` trx id 2A8BC lock mode S waiting
Record lock, heap no 3 PHYSICAL RECORD: n_fields 2; compact format; info bits 32
 0: len 4; hex 00000002; asc ;;
 1: len 4; hex 00000002; asc ;;
*** WE ROLL BACK TRANSACTION (1)

analyze

Reading the deadlock log

When encountering a deadlock, the first step is to read the deadlock log. The deadlock log is usually divided into two parts. The first part shows what lock transaction 1 is waiting for:

170219 13:31:31
*** (1) TRANSACTION:
TRANSACTION 2A8BD, ACTIVE 11 sec starting index read
mysql tables in use 1, locked 1
LOCK WAIT 2 lock struct(s), heap size 376, 1 row lock(s)
MySQL thread id 448218, OS thread handle 0x2abe5fb5d700, query id 18923238 renjun.fangcloud.net 121.41.41.92 root updating
delete from test where a = 2
*** (1) WAITING FOR THIS LOCK TO BE GRANTED:
RECORD LOCKS space id 0 page no 923 n bits 80 index `a` of table `oauthdemo`.`test` trx id 2A8BD lock_mode X waiting
Record lock, heap no 3 PHYSICAL RECORD: n_fields 2; compact format; info bits 32
 0: len 4; hex 00000002; asc ;;
 1: len 4; hex 00000002; asc ;;

From the log, we can see that transaction 1 is currently executing delete from test where a = 2 This statement is applying for an X lock on index a, so lock_mode X waiting is displayed.

The bottom half of the log then shows the locks that transaction 2 currently holds and is waiting for:

*** (2) TRANSACTION:
TRANSACTION 2A8BC, ACTIVE 18 sec inserting
mysql tables in use 1, locked 1
4 lock struct(s), heap size 1248, 3 row lock(s), undo log entries 2
MySQL thread id 448217, OS thread handle 0x2abe5fd65700, query id 18923239 renjun.fangcloud.net 121.41.41.92 root update
insert into test (id,a) values ​​(10,2)
*** (2) HOLDS THE LOCK(S):
RECORD LOCKS space id 0 page no 923 n bits 80 index `a` of table `oauthdemo`.`test` trx id 2A8BC lock_mode X locks rec but not gap
Record lock, heap no 3 PHYSICAL RECORD: n_fields 2; compact format; info bits 32
 0: len 4; hex 00000002; asc ;;
 1: len 4; hex 00000002; asc ;;
*** (2) WAITING FOR THIS LOCK TO BE GRANTED:
RECORD LOCKS space id 0 page no 923 n bits 80 index `a` of table `oauthdemo`.`test` trx id 2A8BC lock mode S waiting
Record lock, heap no 3 PHYSICAL RECORD: n_fields 2; compact format; info bits 32
 0: len 4; hex 00000002; asc ;;
 1: len 4; hex 00000002; asc ;;

From HOLDS THE LOCKS(S) block in the log, we can see that transaction 2 holds an X lock on index a, and it is a record lock. The lock is acquired by the delete statement executed by transaction 2 in step 2. Because this is an equality query based on a unique index in RR isolation mode (Where a = 2), a record lock is requested instead of a next-key lock.

From WAITING FOR THIS LOCK TO BE GRANTED , which is a shared lock. The lock is applied by the statement insert into test (id,a) values ​​(10,2). Under normal circumstances, the insert statement will apply for an exclusive lock, that is, an X lock, but an S lock appears here. This is because field a is a unique index, so the insert statement will perform a duplicate key check before inserting. In order for this check to succeed, it is necessary to apply for an S lock to prevent other transactions from modifying field a.

So why does the S lock fail? This means that applications for locks on the same field need to queue. There is an unsuccessful X lock before the S lock, so the S lock must wait, thus forming a circular waiting and a deadlock occurs.

By reading the deadlock log, we can clearly know what kind of circular wait is formed by the two transactions. After further analysis, we can reversely infer the cause of the circular wait, that is, the cause of the deadlock.

Deadlock formation flow chart

In order to help you better understand the causes of deadlock, we will explain the process of deadlock formation in the form of a table:

step Transaction 1 Transaction 2
1 begin
2 delete from test where a = 2; The execution is successful. Transaction 2 holds the X lock under a=2, and the type is record lock.
3 begin
4 delete from test where a = 2; Transaction 1 wants to apply for an X lock under a=2, but since transaction 2 has already applied for an X lock and the two X locks are mutually exclusive, the X lock application enters the lock request queue.
5 A deadlock occurs, and transaction 1 has a smaller weight, so it is chosen to be rolled back (becomes the victim). insert into test (id, a) values ​​(10, 2); Since a field has a unique index, an S lock is required to check for duplicate keys. Since the value of a inserted is still 2, it is ranked after the X lock. However, the previous X lock application can only succeed after transaction 2 is committed or rolled back, which forms a circular wait and causes a deadlock.

expand

During the deadlock troubleshooting process, a colleague discovered that the above scenario would produce another deadlock, which could not be reproduced manually and could only be reproduced in a high-concurrency scenario.

The log corresponding to this deadlock is not posted here. lock_mode X locks gap before rec insert intention waiting gap before rec insert intention waiting.

We still use a table to explain in detail the process of deadlock generation:

step Transaction 1 Transaction 2
1 begin
2 delete from test where a = 2; The execution is successful. Transaction 2 holds the X lock under a=2, and the type is record lock.
3 begin
4 [Insert Phase 1] insert into test (id, a) values ​​(10, 2); Transaction 2 applies for an S lock to check the duplicate key. The check was successful.
5 delete from test where a = 2; Transaction 1 wants to apply for an X lock under a=2, but since transaction 2 has already applied for an X lock and the two X locks are mutually exclusive, the X lock application enters the lock request queue.
6 A deadlock occurs, and transaction 1 has a smaller weight, so it is chosen to be rolled back (becomes the victim). [Insert Phase 2] insert into test (id, a) values ​​(10, 2); Transaction 2 starts inserting data, and the S lock is upgraded to the X lock, with the type being insert intention. Similarly, X locks enter the queue, forming a circular waiting cycle and causing deadlock.

Summarize

When troubleshooting deadlocks, you first need to analyze the circular wait scenario based on the deadlock log, then analyze the lock type and sequence based on the SQL executed by each current transaction, and reversely infer how the circular wait is formed. In this way, you can find the cause of the deadlock.

Well, that’s all for this article. I hope that the content of this article can be of some help to your study or work. The above analysis is based on experience. I hope that other friends can point out the errors and deficiencies. Thank you for your support for 123WORDPRESS.COM.

You may also be interested in:
  • Detailed explanation of deadlock situations in MySQL and how to deal with deadlocks
  • MySQL database deadlock causes and solutions
  • How to find the ID of a deadlock in a MySQL thread
  • Analysis of a deadlock instance in MySQL database
  • Analysis of a MySQL deadlock scenario example
  • A detailed discussion of MySQL deadlock and logs
  • MYSQL performance optimization sharing (sharding of databases and tables)
  • MyBatis implements Mysql database sub-library and sub-table operations and summary (recommended)
  • Summary of MYSQL database data splitting: sub-library and sub-table
  • Detailed explanation of MySQL deadlock and database and table sharding issues

<<:  Vue shuttle box realizes up and down movement

>>:  jQuery uses hide() and toggle() functions to realize the camera brand display hiding function

Recommend

JavaScript implements displaying a drop-down box when the mouse passes over it

This article shares the specific code of JavaScri...

Solve the problem of docker's tls (ssl) certificate expiration

Problem phenomenon: [root@localhost ~]# docker im...

js uses FileReader to read local files or blobs

Table of contents FileReader reads local files or...

Example of using CSS filter to write mouse over effect

Use CSS filter to write mouse over effect <div...

Detailed explanation of Vue development Sort component code

Table of contents <template> <ul class=&...

Detailed explanation of the solution to image deformation under flex layout

Flex layout is a commonly used layout method nowa...

Tips and precautions for using MySQL index

1. The role of index In general application syste...

Introduction to CSS3 color value RGBA and gradient color usage

Before CSS3, gradient images could only be used a...

Detailed explanation of mysql5.6 master-slave setup and asynchronous issues

Table of contents 1. MySQL master-slave replicati...

Detailed explanation of MySql 5.7.17 free installation configuration tutorial

1. Download the mysql-5.7.17-winx64.zip installat...

The marquee element implements effects such as scrolling fonts and pictures

The marquee element can achieve simple font (image...

How to use stored procedures in MySQL to quickly generate 1 million records

Preface When testing, in order to test the projec...

Detailed explanation of HTML tables

Function: data display, table application scenari...

Differences between MySQL MyISAM and InnoDB

the difference: 1. InnoDB supports transactions, ...

Let's take a look at some powerful operators in JavaScript

Table of contents Preface 1. Null coalescing oper...