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%、函数计算)。
  • 定期检查索引使用情况,删除冗余索引。
相关推荐
未来之窗软件服务33 分钟前
solidwors插件 开发————仙盟创梦IDE
前端·javascript·数据库·ide·仙盟创梦ide
yc_122434 分钟前
SqlHelper 实现类,支持多数据库,提供异步操作、自动重试、事务、存储过程、分页、缓存等功能。
数据库·c#
Leo.yuan2 小时前
基于地图的数据可视化:解锁地理数据的真正价值
大数据·数据库·信息可视化·数据挖掘·数据分析
好吃的肘子2 小时前
MongoDB入门
数据库·mongodb
noravinsc2 小时前
人大金仓数据库 与django结合
数据库·python·django
代码配咖啡2 小时前
《Navicat之外的新选择:实测支持国产数据库的SQLynx核心功能解析》
数据库
懒大王爱吃狼3 小时前
怎么使用python进行PostgreSQL 数据库连接?
数据库·python·postgresql
时序数据说3 小时前
IoTDB集群的一键启停功能详解
大数据·数据库·开源·时序数据库·iotdb
小叶子来了啊3 小时前
信息系统运行管理员:临阵磨枪版
运维·服务器·数据库
数据库幼崽4 小时前
MySQL 8.0 OCP 1Z0-908 131-140题
数据库·mysql·ocp