19 | 为什么我只查一行的语句,也执行这么慢?
git-zjx opened this issue · comments
第一类:查询长时间不返回
一般碰到这种情况的话,大概率是表 t 被锁住了。分析原因时一般都是首先执行一下 show processlist
命令,看看当前语句处于什么状态
等 MDL 锁
出现这个状态表示的是,现在有一个线程正在表 t 上请求或者持有 MDL 写锁,把 select 语句堵住了。
这类问题的处理方式,就是找到谁持有 MDL 写锁,然后把它 kill 掉。通过查询 sys.schema_table_lock_waits 这张表,我们就可以直接找出造成阻塞的 process id,把这个连接用 kill 命令断开即可(MySQL 启动时需要设置 performance_schema=on,相比于设置为 off 会有 10% 左右的性能损失)
等 flush
这个状态表示的是,现在有一个线程正要对表 t 做 flush 操作。也可能是有一个 flush tables 命令被别的语句堵住了,然后它又堵住了我们的 select 语句
这类问题的处理方式,可以执行 show processlist
查看结果,然后 kill 掉堵塞的语句
等行锁
这个问题并不难分析,但问题是怎么查出是谁占着这个写锁。如果你用的是 MySQL 5.7 版本,可以通过 sys.innodb_lock_waits
表查到。
mysql> select * from sys.innodb_lock_waits where locked_table='`test`.`t`'\G
可以看到,这个信息很全,4 号线程是造成堵塞的罪魁祸首。而干掉这个罪魁祸首的方式,就是 KILL QUERY 4
或 KILL 4
。不过,这里不应该显示 KILL QUERY 4
。这个命令表示停止 4 号线程当前正在执行的语句,而这个方法其实是没有用的。因为占有行锁的是 update 语句,这个语句已经是之前执行完成了的,现在执行 KILL QUERY
,无法让这个事务去掉 id=1 上的行锁。实际上,KILL 4
才有效,也就是说直接断开这个连接。这里隐含的一个逻辑就是,连接被断开的时候,会自动回滚这个连接里面正在执行的线程,也就释放了 id=1 上的行锁。
第二类:查询慢
没有索引
mysql> select * from t where c=50000 limit 1;
由于字段 c 上没有索引,这个语句只能走 id 主键顺序扫描,因此需要扫描 5 万行,随着数据量的增加,执行的时间会越来越长,坏查询不一定是慢查询
回滚日志太大
假如生成了 100w 条回滚日志,如果是一致性读的话,会依次执行 undo log,执行 100w 次之后才把结果返回