mysql出现了waiting for global read lock 全局事务等待问题排查

问题发现

在一个普通的日子里,寒澈在操作他负责的一个内部系统的时候,突然发现一个修改状态的操作不能操作了。 查看服务器日志,发现数据库连接超时了:
mysql连接超时
第一反应是数据库挂了,导致数据库连不上了。

可是,神奇的是,接下来发现数据库的查询功能还能用,但是所有的状态修改类的写操作都不能用了。 这下知道遇到麻烦了,肯定是mysql数据库的产生锁了,把所有的写操作阻塞了。

按照现在的这个情况,mysql读没有问题,写被阻塞,出问题的锁肯定是读锁(共享锁),而且还是全局性质的,因为所有的写都被影响了。

问题定位

按照上面的思路去数据库里执行一下SHOW ENGINE INNODB STATUS, 查看一下数据库的状态。 结果如下:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
*** WE ROLL BACK TRANSACTION (2)
------------
TRANSACTIONS
------------
Trx id counter 48415098
Purge done for trx's n:o < 48412278 undo n:o < 0 state: running but idle
History list length 0
LIST OF TRANSACTIONS FOR EACH SESSION:
---TRANSACTION 421386947297640, not started
0 lock struct(s), heap size 1128, 0 row lock(s)
---TRANSACTION 421386947296832, not started
0 lock struct(s), heap size 1128, 0 row lock(s)
---TRANSACTION 421386947296024, not started
0 lock struct(s), heap size 1128, 0 row lock(s)
---TRANSACTION 421386947295216, not started
0 lock struct(s), heap size 1128, 0 row lock(s)
---TRANSACTION 421386947294408, not started
0 lock struct(s), heap size 1128, 0 row lock(s)
---TRANSACTION 421386947293600, not started
0 lock struct(s), heap size 1128, 0 row lock(s)
---TRANSACTION 421386947291984, not started
0 lock struct(s), heap size 1128, 0 row lock(s)
---TRANSACTION 421386947288752, not started
0 lock struct(s), heap size 1128, 0 row lock(s)
---TRANSACTION 421386947287944, not started
0 lock struct(s), heap size 1128, 0 row lock(s)
---TRANSACTION 421386947287136, not started
0 lock struct(s), heap size 1128, 0 row lock(s)
---TRANSACTION 421386947283904, not started
0 lock struct(s), heap size 1128, 0 row lock(s)
---TRANSACTION 421386947280672, not started
0 lock struct(s), heap size 1128, 0 row lock(s)

发现了大量的如上的事务在积压,没有执行。 与预期一致,就是某个锁导致事务阻塞了。

注意这里不会是死锁问题,因为死锁是锁冲突,mysql会自动解决的,不会导致事务长时间阻塞。

查询一下mysql的执行进程,看看情况:

1
2
# 查询一个耗时最久的 进程
select * from information_schema.PROCESSLIST order by time desc;

发现了如下场景
waiting for global read lock

由此我们知道:

  1. 确实发生了一个全局性的读锁waiting for global read lock导致了后续写操作的进行。 后面的事务都在等待这个读锁的释放。
  2. 情况比我们想象的严重,这个读锁不仅影响了我们自己的实例,还把整个mysql下所有的库实例都影响了。
  3. 问题的原因也发现了,就是有一个binlog dump后没有结束导致的。也就是那个耗时最久的进程。

问题解决

既然发现了这个锁产生的原因,直接把这个进程kill掉就可以了

1
kill 3518815

binlog的进程杀掉后,后面阻塞的任务果然都没了。

记录一下排查问题过程中常用的sql

1
2
3
4
5
6
7
8
9
10
11
12
13
14
# 查询当前元数据锁及关联的线程和事务
select * from performance_schema.metadata_locks l
left join performance_schema.threads t on t.THREAD_ID = l.OWNER_THREAD_ID
left join performance_schema.events_transactions_current et on et.THREAD_ID = t.THREAD_ID
left join information_schema.INNODB_TRX trx on t.PROCESSLIST_ID = trx.trx_mysql_thread_id

# 当前系统内的事务
select * from information_schema.innodb_trx;

# 查询一个耗时最久的 进程
select * from information_schema.PROCESSLIST order by time desc;

# 查询innodb 状态
SHOW ENGINE INNODB STATUS

问题避免

发现问题,解决问题,避免问题才是我们排查问题最终要的目的。

我们这个问题是flink cdc 数据的时候导致的,后续的解决办法很简单:
第一: 以后flink 不要cdc主库了,避免对业务产生影响。
第二:flink cdc 一定要做好异常拦截处理,我们这就是因为异常导致fink退出了,无法释放锁
第三:如果可以的话,尽量使用flink的无锁cdc版本

关于这个global read lock是怎么产生的

当mysql在dump整个数据库的时候,会对数据库执行一个FLUSH TABLES WITH READ LOCK命令来加锁,这个命令会停止对所有表的写操作,‌并允许读操作继续。
等备份完成,会使用UNLOCK TABLES来解锁,

参考

mysql 加 global read lock 机制


mysql出现了waiting for global read lock 全局事务等待问题排查
https://www.hancher.top/2024/08/01/db-mysql-waiting-for-global-read-lock/
作者
寒澈
发布于
2024年8月1日
许可协议