19 | 为什么我只查一行的语句,也执行这么慢?

以下内容出自 《MySQL 实战 45 讲》

19 | 为什么我只查一行的语句,也执行这么慢?

第一类:查询长时间不返回

mysql> select * from t where id=1;

一般碰到这种情况。大概率是表 t 被锁住了。

分析:

首先执行 show processlist 命令,查看当前语句的状态。

  • 等 MDL 锁

image

如上图,出现了 Waiting for table metadata lock, 这表示现在有一个线程正在表 t 上请求或者持有 MDL 写锁,把 select 语句堵住了。

这类问题的处理方式,就是找到谁持有 MDL 写锁,然后把它 kill 掉。

通过查询 sys.schema_table_lock_waits 这张表,就可以直接找出造成阻塞的 process id,把这个连接用 kill 命令断开即可。

image

  • 等 flush
    image

MySQL 里面对表做 flush 操作的用法,一般有以下两个:

flush tables t with read lock;
flush tables with read lock;

正常这两个语句执行起来都很快,除非它们也被别的线程堵住了。

出现 Waiting for table flush 状态的可能情况是:有一个 flush tables 命令被别的语句堵住了,然后它又堵住了.select 语句。

这种处理方式和上一种一样。

  • 等行锁
mysql> select * from t where id=1 lock in share mode; 

于访问 id=1 这个记录时要加读锁,如果这时候已经有一个事务在这行记录上持有一个写锁, select 语句就会被堵住。查询谁占用着写锁,可以用下面的语句,查询表 sys.innodb_lock_waits

mysql> select * from t sys.innodb_lock_waits where locked_table='`test`.`t`'\G

在这里插入图片描述

4 号线程是造成堵塞的罪魁祸首。 而干掉这个罪魁祸首的方式,就是 KILL QUERY 4 或 KILL 4。

  • KILL QUERY 4:表示停止 4 号线程当前正在执行的语句,而这个方法其实是没有用的。因为占有行锁的是 update 语句。
  • KILL 4:表示直接断开这个连接。连接被断开时,会自动回滚这个连接里面正在执行的线程,同时也就释放了 id = 1 上的行锁。

第二类 查询慢

  • 字段没有索引,走了全表扫描。
  • 当前事务查询的数据被别的事务大量修改,导致一致性读要依次执行 undo log, 才能返回一致性是视图结果。
posted @ 2023-07-06 21:35  LionelYee  阅读(27)  评论(0编辑  收藏  举报