当前位置: 首页 > news >正文

MySQL诊断系列(3/6):索引分析——5个SQL揪出“僵尸索引”

🔗 接上一篇《MySQL锁问题排查全攻略》,今天我们来“清理门户”——找出那些从不工作却白白占用资源的“僵尸索引”。

你有没有想过:

  • 为什么你的表 ALTER TABLE 越来越慢?
  • 磁盘空间莫名增长,却找不到原因?
  • INSERT/UPDATE 性能下降,怀疑是索引拖累?

很可能,你的数据库里藏着一批“僵尸索引”——它们被创建后,从未被查询使用过,却一直在消耗磁盘空间、拖慢写操作。

今天,我就用5个SQL,带你把这些“寄生虫”一个个揪出来!


🧟 什么是“僵尸索引”?

在MySQL中,每创建一个索引,InnoDB就会生成对应的B+树结构,并维护它。这意味着:

  • 写放大:每次 INSERT/UPDATE/DELETE,不仅要改数据页,还要改所有相关索引页。
  • 空间占用:一个索引可能占用几MB甚至几GB空间。
  • 维护成本OPTIMIZE TABLEALTER TABLE 时间变长。

如果一个索引从来没人用它来查询,那它就是纯粹的“负资产”。


🔍 5个SQL,全面扫描索引健康状况

1️⃣ 查找“零使用”索引 —— 最典型的僵尸


SELECTobject_nameAS table_name,index_name,count_starAS usage_count
FROM performance_schema.table_io_waits_summary_by_index_usage
WHERE object_schema = 'your_database'
AND index_name IS NOT NULL AND count_star = 0
AND sum_timer_wait = 0;

✅ 解读:

  • count_star = 0:该索引从未被任何查询使用过。
  • 这类索引可以优先考虑删除

2️⃣ 找出“最忙”索引 —— 真正的功臣


SELECTobject_nameAS table_name,index_name,(sum_timer_wait / 1000000000000)AS total_wait_sec
FROM performance_schema.table_io_waits_summary_by_index_usage
WHERE object_schema = 'your_database'
AND index_name IS NOT NULL ORDER BY sum_timer_waitDESC
LIMIT 10;

✅ 解读:

  • 这些是数据库的“劳模索引”,使用频率最高。
  • 重点保护对象,不要轻易改动。

3️⃣ 识别“全表扫描”重灾区 —— 缺少索引的表


SELECTobject_nameAS table_name,(sum_timer_wait / 1000000000000)AS full_scan_wait_sec
FROM performance_schema.table_io_waits_summary_by_index_usage
WHERE object_schema = 'your_database'
AND index_nameISNULL  -- 没有使用索引(即全表扫描)
AND sum_timer_wait > 30000000000000  -- 等待时间 > 30秒
ORDER BY sum_timer_wait DESC
LIMIT 5;

🔥 警报:

这些表很可能缺少关键索引,导致频繁全表扫描,是性能瓶颈的源头!


4️⃣ 查看索引大小 —— 空间占用大户


SELECTt.table_name,s.index_name,ROUND(s.stat_value * @@innodb_page_size / 1024 / 1024, 2)AS index_size_mb
FROM mysql.innodb_index_stats s
JOIN information_schema.tables t
ON s.table_name = t.table_name AND s.database_name = t.table_schema
WHERE s.database_name = 'your_database'
AND s.stat_name = 'size'
AND s.index_name IS NOT NULL ORDER BY s.stat_value DESC;

💡 技巧:

结合“零使用”查询,如果一个索引又大又不用,删除它带来的收益会非常显著


5️⃣ 检查重复/冗余索引 —— “兄弟索引”只需留一个

虽然没有直接SQL,但可以通过分析 SHOW CREATE TABLE 来识别:


SHOW CREATE TABLE your_table;

常见冗余模式:

  • KEY idx_a (a), KEY idx_a_b (a,b)idx_a 是冗余的
  • KEY idx_b_a (b,a), KEY idx_a_b (a,b) → 顺序不同,但可能有重叠作用

✅ 建议:保留更通用的复合索引,删除单列索引。


✅ 删除索引的正确姿势

  1. 先观察:确认该索引至少7天内未被使用。

  2. 备份CREATE INDEX 语句记下来,万一误删可快速恢复。

  3. 低峰期操作:使用 DROP INDEX

    
    ALTER TABLE your_table DROP INDEX idx_name;
  4. 监控影响:删除后观察写性能是否有提升。


📣 总结

索引不是越多越好,合理的索引才是好索引

定期执行这5个SQL,可以:

  • 🔍 发现并删除“僵尸索引”,释放空间
  • ⚡ 识别“全表扫描”表,补充缺失索引
  • 📉 降低写操作延迟,提升整体性能

🔗 下期预告:

下一篇《MySQL性能瓶颈定位:缓冲池、临时表与I/O等待》,我们将深入数据库的“心脏”——内存与磁盘,找出性能瓶颈的根源!

📌 点赞 + 关注,解锁更多数据库秘籍!

👉 你的MySQL“瘦身计划”,从今天开始!

PS: 看完是不是觉得要记下好多的SQL,排查步骤又繁琐,不要担心,在 AI 的时代,让大模型来替我们排查分析数据库问题,推荐一款开源好用的MCP Server 工具:SmartDB_MCP ,它不仅能让AI与多种数据库“畅聊无阻”,还能像瑞士军刀一样,提供从SQL优化到数据库健康检测分析的一站式解决方案。
github地址 : https://github.com/wenb1n-dev/SmartDB_MCP
博文地址:SmartDB:AI与数据库的“翻译官”,开启无缝交互新时代!

http://www.dtcms.com/a/340307.html

相关文章:

  • Docker Compose命令一览(Docker Compose指令、docker-compose命令)
  • 动态规划----8.乘积最大子数组
  • 遥感机器学习入门实战教程|Sklearn 案例④ :多分类器对比(SVM / RF / kNN / Logistic...)
  • 详解 scikit-learn 数据预处理工具:从理论到实践
  • 5.4 4pnpm 使用介绍
  • 给你的Unity编辑器添加实现类似 Odin 的 条件显示字段 (ShowIf/HideIf) 功能
  • Scikit-learn 预处理函数分类详解
  • pnpm : 无法加载文件 C:\Program Files\nodejs\pnpm.ps1,因为在此系统上禁止运行脚本。
  • 在 React 中,​父子组件之间的通信(传参和传方法)
  • scikit-learn/sklearn学习|变量去中心化和标准化
  • 2.3 Flink的核心概念解析
  • 详解flink java table api基础(三)
  • Flink Stream API - 顶层Operator接口StreamOperator源码超详细讲解
  • OSPF 典型组网
  • CISP-PTE之路--10文
  • 公有地址和私有地址
  • 【GPT入门】第51课 将hf模型转换为GGUF
  • 深入(流批【牛批】框架)Flink的机制
  • 【Java后端】Spring Boot 全局异常处理最佳实践
  • ssl代理
  • 一会儿能ping通一会ping不通解决方案
  • JavaScript手录18-ajax:异步请求与项目上线部署
  • AI 自动化编程 trae 体验 页面添加富编辑器
  • (5)软件包管理器 yum | Vim 编辑器 | Vim 文本批量化操作 | 配置 Vim
  • 深度解析:RESTful API中的404错误 - 不是所有404都是Bug
  • Vue 3项目中的路由管理和状态管理系统
  • 【Day 31】Linux-LNMP
  • MySQL基础操作
  • SpringBoot + MyBatis-Plus 使用 listObjs 报 ClassCastException 的原因与解决办法
  • Rabbit 实战指南-学习笔记