MySQL索引使用一定有效吗?如何排查索引效果?

MySQL索引使用一定有效吗?如何排查索引效果?

1. 索引一定有效吗?

不一定! 即使你创建了索引,MySQL 也可能因为以下原因 不使用索引索引效果不佳

  • 索引选择错误:MySQL 优化器可能选择了错误的索引。
  • 索引失效场景:某些 SQL 写法会导致索引失效。
  • 数据分布问题:数据量太少或数据分布不均,导致全表扫描更快。
  • 索引设计不合理:索引列顺序、类型不匹配等。

2. 索引失效的常见场景

(1) 使用 !=NOT INNOT EXISTS

复制代码
SELECT * FROM users WHERE age != 20;  -- 可能不走索引
SELECT * FROM users WHERE id NOT IN (1, 2, 3);  -- 可能全表扫描

(2) 使用 LIKE 以通配符开头

复制代码
SELECT * FROM users WHERE name LIKE '%Alice%';  -- 不走索引
SELECT * FROM users WHERE name LIKE 'Alice%';   -- 可能走索引

(3) 对索引列使用函数或计算

复制代码
SELECT * FROM users WHERE YEAR(create_time) = 2023;  -- 不走索引
-- 优化:使用范围查询
SELECT * FROM users WHERE create_time BETWEEN '2023-01-01' AND '2023-12-31';

(4) 数据类型不匹配(隐式转换)

复制代码
-- name 是 VARCHAR,但用数字查询
SELECT * FROM users WHERE name = 123;  -- 不走索引(隐式转换成字符串)

(5) 复合索引未遵循最左前缀原则

复制代码
-- 假设有联合索引 (name, age)
SELECT * FROM users WHERE age = 20;  -- 不走索引(缺少 name 条件)

(6) 数据量太少

  • 如果表只有几十行数据,MySQL 可能直接全表扫描,因为索引查找+回表的开销更大。

3. 如何排查索引效果?

方法 1:使用 EXPLAIN 分析执行计划

复制代码
EXPLAIN SELECT * FROM users WHERE name = 'Alice';

重点关注:

  • type :查询类型(const > ref > range > index > ALL
  • key:实际使用的索引
  • rows:预估扫描行数
  • Extra :额外信息(Using index 表示覆盖索引)

示例输出:

id select_type table type key rows Extra
1 SIMPLE users ref idx_name 1 NULL
  • type=ALL:全表扫描(索引可能未生效)
  • key=NULL:未使用索引

方法 2:检查索引使用情况

复制代码
-- 查看表的索引
SHOW INDEX FROM users;

-- 查看索引使用统计(需开启性能模式)
SELECT * FROM performance_schema.table_io_waits_summary_by_index_usage 
WHERE OBJECT_SCHEMA = 'your_db' AND OBJECT_NAME = 'users';
  • index_name:索引名称
  • count_read:索引被读取次数(越高说明索引越有效)

方法 3:使用 OPTIMIZER_TRACE 查看优化器决策

复制代码
-- 开启优化器跟踪
SET optimizer_trace = 'enabled=on';
SET optimizer_trace_max_mem_size = 1000000;

-- 执行查询
SELECT * FROM users WHERE name = 'Alice';

-- 查看优化器决策
SELECT * FROM information_schema.optimizer_trace;
  • 可以看到 MySQL 为什么选择(或不选择)某个索引。

方法 4:强制使用索引(测试对比)

复制代码
-- 强制使用索引
SELECT * FROM users FORCE INDEX (idx_name) WHERE name = 'Alice';

-- 对比性能
EXPLAIN SELECT * FROM users FORCE INDEX (idx_name) WHERE name = 'Alice';
EXPLAIN SELECT * FROM users IGNORE INDEX (idx_name) WHERE name = 'Alice';
  • 如果强制索引后查询变快,说明优化器可能选错了索引。

4. 如何优化索引?

(1) 选择合适的索引列

  • 高选择性列 (如 user_idgender 更适合索引)。
  • 频繁查询的列 (如 WHEREORDER BYJOIN 条件)。

(2) 使用覆盖索引

复制代码
-- 优化前:需要回表
SELECT id, name, age FROM users WHERE name = 'Alice';

-- 优化后:使用 (name, age) 联合索引,避免回表
ALTER TABLE users ADD INDEX idx_name_age (name, age);

(3) 避免索引冗余

复制代码
-- 已有 (name, age) 索引,再建 (name) 就是冗余的
ALTER TABLE users DROP INDEX idx_name;

(4) 定期分析表

复制代码
-- 更新索引统计信息
ANALYZE TABLE users;

-- 重建索引(修复碎片化)
OPTIMIZE TABLE users;

5. 总结

问题 解决方案
索引未生效 检查 EXPLAIN,避免索引失效场景
优化器选错索引 使用 FORCE INDEXOPTIMIZER_TRACE 分析
索引效果差 优化索引设计,使用覆盖索引
数据量太少 可能不需要索引

📌 建议:

  • 使用 EXPLAIN 分析关键查询。
  • 避免索引失效写法(如 LIKE '%xxx%、函数计算)。
  • 定期检查索引使用情况,删除冗余索引。
相关推荐
小光学长13 分钟前
基于vue框架的电信用户业务管理系统的设计与实现8ly70(程序+源码+数据库+调试部署+开发环境)带论文文档1万字以上,文末可获取,系统界面在最后面。
数据库
程序员不想YY啊28 分钟前
MySQL元数据库完全指南:探秘数据背后的数据
数据库·mysql·oracle
数据最前线32 分钟前
Doris表设计与分区策略:让海量数据管理更高效
数据库
时光追逐者41 分钟前
MongoDB从入门到实战之MongoDB快速入门(附带学习路线图)
数据库·学习·mongodb
头顶秃成一缕光1 小时前
Redis的主从模式和哨兵模式
数据库·redis·缓存
AIGC大时代1 小时前
高效使用DeepSeek对“情境+ 对象 +问题“型课题进行开题!
数据库·人工智能·算法·aigc·智能写作·deepseek
博睿谷IT99_1 小时前
数据库证书可以选OCP认证吗?
数据库·oracle·开闭原则·ocp认证
乐维_lwops1 小时前
数据库监控 | MongoDB监控全解析
数据库·mongodb·数据库监控
观无1 小时前
Redis安装及入门应用
数据库·redis·缓存
柏油2 小时前
MySql InnoDB 事务实现之 undo log 日志
数据库·后端·mysql