`

数据库优化:Waiting for table metadata lock的解决

 
阅读更多

转载自http://blog.csdn.net/qiuyepiaoling/article/details/7821984

 

一个简单的DDL操作(表数据很少,访问高)执行很久,导致Waiting for table metadata lock。

版本:mysql5.5.17

查看:Innodb_buffer_pool_pages_free = 0

解决:set global innodb_stats_on_metadata=0

查看丁奇老大博客:原文http://dinglin.iteye.com/blog/1575840

 

这个问题来自冷之同学测试时候碰到的一个“诡异现象”。

 

1、 测试现象       

测试的库有很多数据,但是重启之后,只对一个表的5w条记录作查询。查询条件客户端控制,确保查询范围。innodb_buffer_pool_size设置为35G。

现象1:查询性能会出现大幅度抖动;

现象2:介入追查后发现,Innodb_buffer_pool_pages_free = 0

 

         其中bp剩余量这个是最直观异常的,因为访问的5w行记录撑死也不可能把35G内存吃光的。在QA同学确认没有别人在使用这个库的情况下。

 

2、过程和原因

         其实几乎确定还是有别的查询在访问的。所以打开general_log。 发现除了QA同学压的语句外,这个Server上还有一些监控语句。

         其中一个语句如下

select constraint_schema,table_name,constraint_name,constraint_type from information_schema.table_constraints where table_schema not in ('information_schema', 'mysql', 'test');  

         这个语句访问了表 information_schema.table_constraints.

跟踪发现这个语句触发了读盘操作。原因是需要访问引擎的info()接口,而InnoDB此时又“顺手”做了更新索引统计的操作dict_update_statistics。

更新索引统计的基本流程是随机读取部分demo行。所以这个操作实际上是访问了这个Server里面的所有表,因此不只是访问5w行

而且由于别的表事先没有被访问,就会导致读盘操作,也包括BP的LRU更新。

 

3、哪些表会触发

         不只是上面提到的table_constraints,information_schema库下的一下几个表,访问时候都会触发这个“顺手”操作。

information_schema.TABLES

information_schema.STATISTICS

information_schema.PARTITIONS

information_schema.KEY_COLUMN_USAGE

information_schema.TABLE_CONSTRAINTS

information_schema.REFERENTIAL_CONSTRAINTS

         其实还有 show table status ,也会触发这个操作,只是只处理单表,所以影响没那么明显。

 

4、修改

头痛医头的方法是把这些监控去掉。但实际上像TABLES、TABLE_CONSTRAINTS这些表,都是静态数据,访问时不作索引统计也没关系的。

另外一个方法就是把innodb_stats_on_metadata设置成off,这样上述说到的这些表访问都不会触发索引统计。

         实际上这个动态统计的功能已经不推荐了,官方已经在6.0以后增加参数控制DML期间也不作动态统计了。因此这个参数配置成off更合理些(默认是on)

分享到:
评论

相关推荐

    MySQL出现Waiting for table metadata lock的原因方法

    而且,一旦alter table TableA的操作停滞在Waiting for table metadata lock的状态,后续对TableA的任何操作(包括读)都无法进行,因为他们也会在Opening tables的阶段进入到Waiting for table metadata lock的锁...

    mysql锁表解表

    而且,一旦alter table TableA的操作停滞在Waiting for table metadata lock的状态,后续对TableA的任何操作(包括读)都无法进行,因为他们也会在Opening tables的阶段进入到Waiting for table metadata lock的锁...

    MySQL表结构变更你不可不知的Metadata Lock详解

    想必玩过mysql的人对Waiting for table metadata lock肯定不会陌生,一般都是进行alter操作时被堵住了,导致了我们在show processlist 时,看到线程的状态是在等metadata lock。本文会对MySQL表结构变更的Metadata ...

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

    #### 状态:Waiting for table metadata lock 当查询长时间不返回时,可能是由于表级元数据锁(Metadata Lock, MDL)导致的。MDL锁用于保护表结构的安全,确保在结构变更期间数据的一致性。当一个线程试图获取MDL锁...

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

    这种情况通常表现为在`SHOW PROCESSLIST`命令中看到查询状态为“Waiting for table metadata lock”。这表明另一个线程正在请求或持有表的MDL(Metadata Lock)写锁,从而阻止了读取操作,如简单的`SELECT`语句。在...

    Mysql解决USE DB堵塞详解

    2. `DROP TABLE A` - 在等待`Waiting for table metadata lock`,这意味着想要删除表A,但无法获取到相应的MDL锁。 3. `SELECT * FROM A` - 同样处于`Waiting for table metadata lock`状态,表示查询A表时,需要的...

    解决sql挂起的代码帮助文档

    SHOW FULL PROCESSLIST WHERE State = 'Waiting for table metadata lock'; -- 强制结束指定ID的进程 KILL [process_id]; ``` 请注意,以上操作应谨慎执行,尤其是KILL命令,因为它可能会导致未保存的数据丢失或...

    MySQL运维-故障处理手册

    1. **Waiting for table metadata lock:** - **含义:**表明当前线程正在等待获取表的元数据锁。 - **解决方法:** - 检查是否有其他线程正在对同一张表进行结构修改(如ALTER TABLE)。 - 尽量减少对表结构的...

    详细分析mysql MDL元数据锁

    当你看到 waiting for table metadata lock 时,那就是遇到MDL元数据锁了。本篇文章将会介绍MDL锁的产生与排查过程。 1.什么是MDL锁 MDL全称为metadata lock,即元数据锁。MDL锁主要作用是维护表元数据的数据一致性...

    mysql show processlist 显示mysql查询进程

    7. **State**:这是线程执行SQL语句的具体阶段,例如Waiting for table metadata lock(等待表元数据锁)、Sending data(发送数据)或Locked(锁定)。这些状态可以帮助分析查询的执行流程。 8. **Info**:展示...

    mysql45讲 19.为什么我只查一行的语句,也执行这么慢?1

    使用 show processlist 命令查看 Waiting for table metadata lock 的示意图。 这个状态表示的是,现在有一个线程正在表 t 上请求或者持有 MDL 写锁,把 select 语句堵住了。在第 6 篇文章《全局锁和表锁:给表加个...

    mysql-常见问题,索引优化

    `STATE`列显示了每个线程当前的活动状态,如`Waiting for table metadata lock`、`Sending data`等,有助于诊断阻塞和性能问题。 17. 松散索引和紧凑索引 松散索引(也称为非聚簇索引)不包含主键数据,只存储索引...

    MySQL如何查看元数据锁阻塞在哪里

    如果某个会话处于等待状态(Waiting for table metadata lock),这表明它正在等待另一个会话释放元数据锁。 2. 通过`information_schema.innodb_trx`表:这个内部表包含了所有活跃的InnoDB事务信息。通过查询此表...

    通过mysql show processlist 命令检查mysql锁的方法

    7. **State**:线程执行SQL语句的具体状态,如Waiting for table metadata lock(等待表元数据锁)或Sending data(发送数据)等,这些状态提供了关于查询执行进度的线索。 8. **Info**:显示正在执行的SQL语句,但...

    processlist命令 查看mysql 线程

    - `State`列中的各种状态可以帮助识别潜在的问题,比如长时间的`Waiting for table metadata lock`可能表明有锁定问题,`Copying to tmp table on disk`可能表示内存不足,导致数据需写入磁盘临时表。 5. **优化和...

Global site tag (gtag.js) - Google Analytics