0

0

mysql覆盖索引是什么_mysql索引优化与回表分析

P粉602998670

P粉602998670

发布时间:2026-02-28 10:43:49

|

794人浏览过

|

来源于php中文网

原创

覆盖索引是查询所需字段全部被同一索引包含的状态,可避免回表;判断依据是explain中extra显示using index,而非using index condition或using where。

mysql覆盖索引是什么_mysql索引优化与回表分析

覆盖索引就是“查什么,索引里就有什么”

覆盖索引不是一种特殊索引类型,而是**查询与索引列完全匹配的一种状态**:当 SELECT 的所有字段、WHERE 条件字段、ORDER BYGROUP BY 字段,全部被包含在同一个索引的列中时,MySQL 就能直接从二级索引叶子节点取到全部数据,跳过回表操作。

关键判断依据是 EXPLAIN 输出中的 Extra 列是否显示 Using index —— 出现这个标记,说明真正用上了覆盖索引;如果显示 Using index condition,只是用了索引下推(ICP),仍需回表。

  • 只查索引列:比如 SELECT id, age FROM user WHERE age = 25,而索引是 INDEX idx_age (age) → ❌ 不覆盖(缺 id
  • 联合索引对齐:改成 INDEX idx_age_id (age, id) → ✅ 覆盖(WHERE + SELECT 全在索引里)
  • 注意隐式排序字段:如果 ORDER BY age,且 age 是联合索引最左列,也能复用索引排序,避免文件排序(Using filesort

为什么回表慢?不只是“多一次IO”那么简单

回表的本质是:先走二级索引树拿到一批无序主键值,再拿着这些主键去聚簇索引里逐个查找——这会触发大量**随机磁盘 IO**。因为二级索引叶子节点里的主键值通常是乱序的(比如查出主键 102、7、883、45),导致 MySQL 要反复加载不同页(page)进 buffer pool,每次加载都可能淘汰有用缓存、引发锁竞争、拖慢并发吞吐。

尤其在大结果集场景下,回表开销可能超过全表扫描。例如百万级订单表按 status 查询并 SELECT *,即使有 INDEX idx_status(status),性能也常不如扫聚簇索引本身。

Daft Art
Daft Art

AI专辑封面图片生成器

下载
  • 回表 ≠ 一定慢:小结果集(
  • MRR(Multi-Range Read)可缓解:开启后会对回表主键预排序,把随机 IO 变成顺序 IO,但依赖 read_rnd_buffer_size 配置和优化器成本估算(mrr_cost_based=on 默认启用)
  • 别迷信“索引越多越好”:覆盖索引列越多,索引体积越大,写入更新越慢,尤其是高频率 UPDATE 的字段要慎加进覆盖索引

怎么设计真正有效的覆盖索引?

不是把所有常用字段堆进一个联合索引就行。得按「查询驱动」来反向建模:先看慢查询的 SELECTWHEREORDER BYLIMIT 模式,再按最左前缀原则排列字段顺序。

典型错误是把过滤低频字段放最左,比如 INDEX (create_time, user_id, status) 用于 WHERE status = 'paid' —— 因为 status 不是最左,该索引根本无法命中。

  • 高频等值条件放最左:如经常查 WHERE tenant_id = ? AND status = ?,索引应为 (tenant_id, status)
  • 范围查询字段放最后:比如 WHERE a = 1 AND b > 100 AND c = 2,索引优先 (a, c, b),而非 (a, b, c)b 后面的字段无法走索引)
  • 避免冗余索引:已有 (a, b, c),就不必再建 (a, b);但 (a, c) 可能仍有价值(覆盖仅查 a/c 的查询)
  • 字符串字段谨慎包含:VARCHAR(255) 全长进索引会大幅增加 B+Tree 层高和内存占用,可用前缀索引(如 name(10))替代,但前缀索引不能用于 ORDER BY name 或覆盖 SELECT name

EXPLAIN 看懂了,但还是没覆盖?常见失效陷阱

即使写了联合索引,也常因隐式类型转换、函数包裹、OR 条件或 SELECT * 导致覆盖失效。这些细节不报错,但悄悄让 Extra 变成 Using where; Using index 或干脆 Using where

  • WHERE age + 1 = 26 → 对字段做运算,索引失效(改用 WHERE age = 25
  • WHERE CAST(create_time AS DATE) = '2025-01-01' → 函数导致无法走索引(改用 WHERE create_time >= '2025-01-01' AND create_time )
  • WHERE status = 'paid' OR amount > 100OR 中任一条件没索引,整条语句可能放弃索引(拆成 UNION 或补全索引)
  • SELECT * FROM user WHERE age = 25 → 即使有 INDEX(age, id, name)* 会引入未索引字段(如 email),强制回表

最易被忽略的一点:覆盖索引对 NULL 值敏感。如果索引列允许 NULL,而查询条件是 WHERE col IS NULL,某些旧版本 MySQL 可能无法高效利用该索引做覆盖 —— 测试时务必用真实数据验证 EXPLAIN 输出。

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
mysql修改数据表名
mysql修改数据表名

MySQL修改数据表:1、首先查看数据库中所有的表,代码为:‘SHOW TABLES;’;2、修改表名,代码为:‘ALTER TABLE 旧表名 RENAME [TO] 新表名;’。php中文网还提供MySQL的相关下载、相关课程等内容,供大家免费下载使用。

682

2023.06.20

MySQL创建存储过程
MySQL创建存储过程

存储程序可以分为存储过程和函数,MySQL中创建存储过程和函数使用的语句分别为CREATE PROCEDURE和CREATE FUNCTION。使用CALL语句调用存储过程智能用输出变量返回值。函数可以从语句外调用(通过引用函数名),也能返回标量值。存储过程也可以调用其他存储过程。php中文网还提供MySQL创建存储过程的相关下载、相关课程等内容,供大家免费下载使用。

432

2023.06.21

mongodb和mysql的区别
mongodb和mysql的区别

mongodb和mysql的区别:1、数据模型;2、查询语言;3、扩展性和性能;4、可靠性。本专题为大家提供mongodb和mysql的区别的相关的文章、下载、课程内容,供大家免费下载体验。

286

2023.07.18

mysql密码忘了怎么查看
mysql密码忘了怎么查看

MySQL是一个关系型数据库管理系统,由瑞典MySQL AB 公司开发,属于 Oracle 旗下产品。MySQL 是最流行的关系型数据库管理系统之一,在 WEB 应用方面,MySQL是最好的 RDBMS 应用软件之一。那么mysql密码忘了怎么办呢?php中文网给大家带来了相关的教程以及文章,欢迎大家前来阅读学习。

519

2023.07.19

mysql创建数据库
mysql创建数据库

MySQL是一个关系型数据库管理系统,由瑞典MySQL AB 公司开发,属于 Oracle 旗下产品。MySQL 是最流行的关系型数据库管理系统之一,在 WEB 应用方面,MySQL是最好的 RDBMS 应用软件之一。那么mysql怎么创建数据库呢?php中文网给大家带来了相关的教程以及文章,欢迎大家前来阅读学习。

264

2023.07.25

mysql默认事务隔离级别
mysql默认事务隔离级别

MySQL是一种广泛使用的关系型数据库管理系统,它支持事务处理。事务是一组数据库操作,它们作为一个逻辑单元被一起执行。为了保证事务的一致性和隔离性,MySQL提供了不同的事务隔离级别。php中文网给大家带来了相关的教程以及文章欢迎大家前来学习阅读。

392

2023.08.08

sqlserver和mysql区别
sqlserver和mysql区别

SQL Server和MySQL是两种广泛使用的关系型数据库管理系统。它们具有相似的功能和用途,但在某些方面存在一些显著的区别。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

539

2023.08.11

mysql忘记密码
mysql忘记密码

MySQL是一种关系型数据库管理系统,关系数据库将数据保存在不同的表中,而不是将所有数据放在一个大仓库内,这样就增加了速度并提高了灵活性。那么忘记mysql密码我们该怎么解决呢?php中文网给大家带来了相关的教程以及其他关于mysql的文章,欢迎大家前来学习阅读。

640

2023.08.14

Golang 测试体系与代码质量保障:工程级可靠性建设
Golang 测试体系与代码质量保障:工程级可靠性建设

Go语言测试体系与代码质量保障聚焦于构建工程级可靠性系统。本专题深入解析Go的测试工具链(如go test)、单元测试、集成测试及端到端测试实践,结合代码覆盖率分析、静态代码扫描(如go vet)和动态分析工具,建立全链路质量监控机制。通过自动化测试框架、持续集成(CI)流水线配置及代码审查规范,实现测试用例管理、缺陷追踪与质量门禁控制,确保代码健壮性与可维护性,为高可靠性工程系统提供质量保障。

0

2026.02.28

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
MySQL 教程
MySQL 教程

共48课时 | 2.4万人学习

MySQL 初学入门(mosh老师)
MySQL 初学入门(mosh老师)

共3课时 | 0.3万人学习

简单聊聊mysql8与网络通信
简单聊聊mysql8与网络通信

共1课时 | 838人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号