0

0

mysql如何根据执行计划进行索引选择与调整

P粉602998670

P粉602998670

发布时间:2026-01-27 12:00:13

|

559人浏览过

|

来源于php中文网

原创

EXPLAIN中key与possible_keys不一致是因为优化器基于成本选择最优索引,而非简单“能用就用”;key只显示最终选定索引,可能因行数预估更少而选时间范围索引而非状态索引。

mysql如何根据执行计划进行索引选择与调整

怎么看 EXPLAIN 输出里的 keypossible_keys 不一致?

MySQL 选索引不是“哪个能用就用哪个”,而是基于成本估算。即使 possible_keys 列出多个索引,key 只显示最终选中的那个——它可能不是你预期的最“宽”或“最左”的索引。

常见误导:看到 possible_keys 包含 idx_user_statusidx_user_created_at,但 key 是后者,就以为“没走状态索引”。其实是因为查询中 created_at > '2024-01-01' 返回行数远少于 status = 'active'(比如后者占全表 60%),优化器判定用时间范围索引更便宜。

  • EXPLAIN FORMAT=JSON 查看 used_range_accessrows 预估,比只看 key 更可靠
  • 如果 possible_keys 为空但实际有索引,检查字段是否用了函数(如 WHERE YEAR(created_at) = 2024)或隐式类型转换(如字符串列和数字比较)
  • key_len 值比索引定义长度小,说明只用了前缀(比如 VARCHAR(255) 字段建了前缀索引 INDEX(name(50))key_len 就是 50 × 字符编码字节数)

复合索引的最左匹配到底卡在哪儿?

“最左前缀原则”不是语法限制,而是 B+ 树检索路径决定的:必须从索引第一列开始连续提供等值条件,才能高效定位数据页。一旦中间出现范围查询(>BETWEENLIKE 'abc%'),后续列就无法用于索引查找,只能用于过滤(Extra: Using index condition)。

SELECT * FROM orders 
WHERE user_id = 123 
  AND status IN ('paid', 'shipped') 
  AND created_at > '2024-05-01';

对这个查询,INDEX(user_id, status, created_at)INDEX(user_id, created_at, status) 更优——因为 status IN 算等值匹配(MySQL 8.0+ 支持多值等值查找),而 created_at > 是范围,放最后才不打断前缀匹配。

  • 排序需求也要纳入索引设计:ORDER BY created_at DESC 跟在等值条件后,可被同一复合索引覆盖,避免 Using filesort
  • SELECT * 时,覆盖索引失效,但若只查索引列(如 SELECT user_id, status),INDEX(user_id, status, created_at) 就能完全避免回表
  • 不要为每个 WHERE 字段单独建索引,优先合并:三个单列索引不如一个设计合理的三列复合索引(除非查询模式差异极大)

typerange 还是 ref为什么影响这么大?

type 直接反映扫描方式:ref 表示通过索引等值查找定位到具体数据页(快),range 表示按索引顺序扫描一段区间(慢,尤其区间大时)。两者性能差距常达数量级。

SpeakingPass-打造你的专属雅思口语语料
SpeakingPass-打造你的专属雅思口语语料

使用chatGPT帮你快速备考雅思口语,提升分数

下载

典型陷阱:明明写了 WHERE category_id = 5 AND price BETWEEN 100 AND 500EXPLAIN 却显示 type: range。原因往往是索引顺序错了——如果建的是 INDEX(price, category_id),优化器只能先按 price 范围扫描,再在结果里过滤 category_id,变成 range;改成 INDEX(category_id, price) 后,category_id = 5 定位到子树,再在该子树内做 price 范围扫描,type 就变成 ref

  • type: ALLindex 是全表/全索引扫描,必须干预;type: consteq_ref 是理想状态(主键/唯一索引等值查找)
  • Extra: Using index 表示覆盖索引,Using where 表示存储引擎返回数据后 Server 层还要过滤——后者意味着索引没包含所有 WHERE 条件列
  • rows 预估值远大于实际返回行数,可能是统计信息过期,执行 ANALYZE TABLE table_name 更新

什么时候该删索引?

索引不是越多越好。写多读少的表,每个额外索引都会拖慢 INSERT/UPDATE/DELETE,还占用内存和磁盘。重点删这三类:

  • 重复索引:INDEX(a,b)INDEX(a) 同时存在,后者无意义
  • 低效索引:SHOW INDEX FROM table_nameCardinality,如果某列唯一值极少(如 is_deleted TINYINT 只有 0/1),且从不单独用于查询,这个单列索引基本无效
  • 长期未用索引:开启 performance_schema,查 sys.schema_unused_indexes 视图(MySQL 5.7+),但注意它只统计 Server 层执行计划记录,不捕获直接命中缓存的查询

调整索引本质是平衡读写开销。一个被高频查询用到的复合索引,即使让单条 UPDATE 慢 5ms,也通常值得;但一个三个月没被 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的相关下载、相关课程等内容,供大家免费下载使用。

667

2023.06.20

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

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

247

2023.06.21

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

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

281

2023.07.18

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

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

515

2023.07.19

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

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

256

2023.07.25

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

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

386

2023.08.08

sqlserver和mysql区别
sqlserver和mysql区别

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

532

2023.08.11

mysql忘记密码
mysql忘记密码

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

601

2023.08.14

Python 自然语言处理(NLP)基础与实战
Python 自然语言处理(NLP)基础与实战

本专题系统讲解 Python 在自然语言处理(NLP)领域的基础方法与实战应用,涵盖文本预处理(分词、去停用词)、词性标注、命名实体识别、关键词提取、情感分析,以及常用 NLP 库(NLTK、spaCy)的核心用法。通过真实文本案例,帮助学习者掌握 使用 Python 进行文本分析与语言数据处理的完整流程,适用于内容分析、舆情监测与智能文本应用场景。

2

2026.01.27

热门下载

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

精品课程

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

共48课时 | 1.9万人学习

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

共3课时 | 0.3万人学习

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

共1课时 | 811人学习

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

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