MySQL 排查全局锁

本文实验参考自《千金良方:MySQL性能优化法则金字塔》performance_schema相关章节

数据库版本:5.7.36单机

sql 复制代码
mysql> select @@version;
+------------+
| @@version  |
+------------+
| 5.7.36-log |
+------------+
1 row in set (0.00 sec)

session 1:

sql 复制代码
mysql> select connection_id();
+-----------------+
| connection_id() |
+-----------------+
|               7 |
+-----------------+
1 row in set (0.00 sec)

session 2:

sql 复制代码
mysql> select connection_id();
+-----------------+
| connection_id() |
+-----------------+
|               8 |
+-----------------+
1 row in set (0.00 sec)

session 3:

sql 复制代码
mysql> select connection_id();
+-----------------+
| connection_id() |
+-----------------+
|               9 |
+-----------------+
1 row in set (0.00 sec)

mysql> show processlist;
+----+------+-----------+-------------+---------+------+----------+------------------+
| Id | User | Host      | db          | Command | Time | State    | Info             |
+----+------+-----------+-------------+---------+------+----------+------------------+
|  7 | root | localhost | backup_test | Sleep   |   37 |          | NULL             |
|  8 | root | localhost | backup_test | Sleep   |    9 |          | NULL             |
|  9 | root | localhost | backup_test | Query   |    0 | starting | show processlist |
+----+------+-----------+-------------+---------+------+----------+------------------+
3 rows in set (0.00 sec)

session 1:开启全局读锁

sql 复制代码
mysql> flush table with read lock;
Query OK, 0 rows affected (0.00 sec)

session 2:修改任意一个表的数据

sql 复制代码
mysql> use backup_test
mysql> select * from full_backup;
+----+----------+-------+
| id | name     | class |
+----+----------+-------+
|  1 | wanger   |     9 |
|  2 | lisan    |     1 |
|  3 | zhangsan |     6 |
|  4 | si       |     2 |
|  5 | wangmazi |     4 |
+----+----------+-------+
5 rows in set (0.00 sec)
mysql> update full_backup set class=3 where id=1; //发现被阻塞

session 3:

sql 复制代码
mysql> show processlist;
+----+------+-----------+-------------+---------+------+------------------------------+-------------------------------------------+
| Id | User | Host      | db          | Command | Time | State                        | Info                                      |
+----+------+-----------+-------------+---------+------+------------------------------+-------------------------------------------+
|  7 | root | localhost | backup_test | Sleep   |  141 |                              | NULL                                      |
|  8 | root | localhost | backup_test | Query   |   49 | Waiting for global read lock | update full_backup set class=3 where id=1 |
|  9 | root | localhost | backup_test | Query   |    0 | starting                     | show processlist                          |
+----+------+-----------+-------------+---------+------+------------------------------+-------------------------------------------+
3 rows in set (0.00 sec)

发现processlist_id=8的会话在等一个全局读锁

使用常规的排查锁的表查询都为空

sql 复制代码
mysql> select * from information_schema.innodb_locks;
Empty set, 1 warning (0.00 sec)

mysql> select * from information_schema.innodb_lock_waits;
Empty set, 1 warning (0.00 sec)

mysql> select * from information_schema.innodb_trx;
Empty set (0.00 sec)

使用show engine innodb status\G;查询事务这块儿也没有什么有用的信息

sql 复制代码
------------
TRANSACTIONS
------------
Trx id counter 6939
Purge done for trx's n:o < 6939 undo n:o < 0 state: running but idle
History list length 0
LIST OF TRANSACTIONS FOR EACH SESSION:
---TRANSACTION 421678777934336, not started
0 lock struct(s), heap size 1136, 0 row lock(s)
---TRANSACTION 421678777933424, not started
0 lock struct(s), heap size 1136, 0 row lock(s)
---TRANSACTION 421678777932512, not started
0 lock struct(s), heap size 1136, 0 row lock(s)
---TRANSACTION 421678777931600, not started
0 lock struct(s), heap size 1136, 0 row lock(s)

查询performance_schema.metadata_locks表发现详情如下

sql 复制代码
mysql> select * from performance_schema.metadata_locks where owner_thread_id!=sys.ps_thread_id(connection_id())\G;
*************************** 1. row ***************************
          OBJECT_TYPE: GLOBAL	#global
        OBJECT_SCHEMA: NULL
          OBJECT_NAME: NULL
OBJECT_INSTANCE_BEGIN: 140202765115488
            LOCK_TYPE: SHARED   #共享锁
        LOCK_DURATION: EXPLICIT	#显示获取
          LOCK_STATUS: GRANTED  #已获取锁
               SOURCE:
      OWNER_THREAD_ID: 33		#thread_id=33
       OWNER_EVENT_ID: 2604
*************************** 2. row ***************************
          OBJECT_TYPE: COMMIT	#commit
        OBJECT_SCHEMA: NULL
          OBJECT_NAME: NULL
OBJECT_INSTANCE_BEGIN: 140202765018160
            LOCK_TYPE: SHARED	#共享锁
        LOCK_DURATION: EXPLICIT	#显示获取
          LOCK_STATUS: GRANTED	#已获取锁
               SOURCE:
      OWNER_THREAD_ID: 33		#thread_id=33
       OWNER_EVENT_ID: 2724
*************************** 3. row ***************************
          OBJECT_TYPE: GLOBAL	#global
        OBJECT_SCHEMA: NULL
          OBJECT_NAME: NULL
OBJECT_INSTANCE_BEGIN: 140202564139616
            LOCK_TYPE: INTENTION_EXCLUSIVE	#意向锁
        LOCK_DURATION: STATEMENT			#通过语句
          LOCK_STATUS: PENDING				#为获取到锁,等待
               SOURCE:
      OWNER_THREAD_ID: 34					#thread_id=34
       OWNER_EVENT_ID: 402
3 rows in set (0.00 sec)

通过上面的查询结果可以看到,thread_id=34的线程在等待thread_id=33的线程持有的锁。

sql 复制代码
mysql> select * from performance_schema.events_statements_current where thread_id=33\G;
*************************** 1. row ***************************
              THREAD_ID: 33
               EVENT_ID: 2598
           END_EVENT_ID: 2728
             EVENT_NAME: statement/sql/flush
                 SOURCE:
            TIMER_START: 523558101594285000
              TIMER_END: 523558104347773000
             TIMER_WAIT: 2753488000
              LOCK_TIME: 0
               SQL_TEXT: flush table with read lock
                 DIGEST: 172f9471494a101656e6cb75da3e2726
            DIGEST_TEXT: FLUSH TABLE WITH READ LOCK
         CURRENT_SCHEMA: backup_test
            OBJECT_TYPE: NULL
          OBJECT_SCHEMA: NULL
            OBJECT_NAME: NULL
  OBJECT_INSTANCE_BEGIN: NULL
            MYSQL_ERRNO: 0
      RETURNED_SQLSTATE: 00000
           MESSAGE_TEXT: NULL
                 ERRORS: 0
               WARNINGS: 0
          ROWS_AFFECTED: 0
              ROWS_SENT: 0
          ROWS_EXAMINED: 0
CREATED_TMP_DISK_TABLES: 0
     CREATED_TMP_TABLES: 0
       SELECT_FULL_JOIN: 0
 SELECT_FULL_RANGE_JOIN: 0
           SELECT_RANGE: 0
     SELECT_RANGE_CHECK: 0
            SELECT_SCAN: 0
      SORT_MERGE_PASSES: 0
             SORT_RANGE: 0
              SORT_ROWS: 0
              SORT_SCAN: 0
          NO_INDEX_USED: 0
     NO_GOOD_INDEX_USED: 0
       NESTING_EVENT_ID: NULL
     NESTING_EVENT_TYPE: NULL
    NESTING_EVENT_LEVEL: 0
1 row in set (0.00 sec)

通过如上查询,可以看到thread_id=33的线程执行的是FLUSH TABLE WITH READ LOCK锁定的,

sql 复制代码
mysql> select processlist_id,thread_id from performance_schema.threads where thread_id=33 or thread_id=34;
+----------------+-----------+
| processlist_id | thread_id |
+----------------+-----------+
|              7 |        33 |
|              8 |        34 |
+----------------+-----------+
2 rows in set (0.01 sec)

查询到线程对应的processlist_id,跟业务进行沟通是否可以进行kill等操作

kill 7;

kill掉之后,session 2的update立马执行成功

sql 复制代码
select sys.ps_thread_id(processlist_id);  --根据processlist_id查出线程号的函数
select connection_id(); --查找当前连接会话的processlist_id;
select sys.ps_thread_id(connection_id());  --查找当前连接会话的线程号

锁排查相关表
select * from information_schema.innodb_locks;
select * from information_schema.innodb_lock_waits;
select * from information_schema.innodb_trx;
相关推荐
czhc114007566341 分钟前
LINUX712 MYSQL;磁盘分区;NFS
数据库·mysql·adb
叁沐1 小时前
MySQL 13 为什么表数据删掉一半,表文件大小不变?
mysql
不太可爱的大白1 小时前
Mysql:分库分表
数据库·mysql
四季豆豆豆1 小时前
博客项目 laravel vue mysql 第四章 分类功能
vue.js·mysql·laravel
十五年专注C++开发2 小时前
hiredis: 一个轻量级、高性能的 C 语言 Redis 客户端库
开发语言·数据库·c++·redis·缓存
憨堡包^—^2 小时前
Docker —— MySQL主从复制集群
mysql·docker·容器
bianguanyue3 小时前
SQLite密码修改故障排查:RSA加密随机性导致的数据库匹配问题
数据库·sqlite·c#
亚马逊云开发者3 小时前
将 Go 应用从 x86 平台迁移至 Amazon Graviton:场景剖析与最佳实践
linux·数据库·golang
凉拌青瓜哈3 小时前
DVWA-LOW级-SQL手工注入漏洞测试(MySQL数据库)+sqlmap自动化注入-小白必看(超详细)
mysql·安全·网络安全
张先shen4 小时前
亿级流量下的缓存架构设计:Redis+Caffeine多级缓存实战
数据库·redis·缓存