mysql索引能提高所有查询性能吗_mysql查询优化误区

1次阅读

索引并非万能,对无条件查询、小表、高重复值字段、like以%开头、联合索引顺序错、范围查询后列等场景无效甚至有害;需结合explain、key列、extra信息综合判断。

mysql索引能提高所有查询性能吗_mysql查询优化误区

索引不是万能的,对某些查询反而拖慢速度

mysql 索引能加速 WHEREORDER BYGROUP BY 和连接字段的查找,但对全表扫描类查询几乎无益,甚至有害。比如 select * FROM user LIMIT 10 这种无条件查询,加了索引反而让优化器多一次 B+ 树遍历,还占额外磁盘和内存。

  • 索引本身需要维护:INSERT/UPDATE/delete 时要同步更新索引树,写入变慢
  • 小表(如 rows )加索引收益极低,优化器常直接走全表扫描
  • 高重复值字段(如 status TINYINT 只有 0/1)建普通索引效果差,SELECT * FROM t WHERE status = 1 可能仍触发全表扫描

LIKE 查询以 % 开头时,索引基本失效

LIKE 是否走索引,关键看通配符位置:name LIKE 'abc%' 可用前缀索引,而 name LIKE '%abc'name LIKE '%abc%' 无法利用 B+ 树的有序结构,只能回表或全扫。

  • 想支持前后模糊查,可考虑 FULLTEXT 索引(仅 MyISAM/InnoDB 的 char/VARCHAR/TEXT 字段)
  • 或改用倒排索引方案(如 elasticsearch),MySQL 原生不擅长
  • 注意 COLLATION 影响:若列是 utf8mb4_0900_as_cs,大小写敏感下 'ABC%' 无法命中 'abc' 索引

联合索引顺序错,等于白建

联合索引 (a, b, c) 的生效前提是「最左前缀匹配」:能用在 a = ?a = ? AND b = ?a = ? AND b = ? AND c = ? 上;但 b = ?b = ? AND c = ? 完全无法使用该索引。

  • 如果查询高频是 WHERE b = ? AND c = ?,应建 (b, c)(b, c, a),而非依赖已有 (a, b, c)
  • ORDER BY 字段也受顺序约束:索引 (a, b) 可覆盖 ORDER BY a, b,但不能覆盖 ORDER BY b, a
  • 区分等值与范围:WHERE a = 1 AND b > 10 AND c = 5 中,c 不会走索引(范围查询后列失效)

EXPLAIN 显示 type=ALL 并不总代表没走索引

type=ALL 表示全表扫描,但有时它只是执行计划的“最终阶段”——比如 SELECT count(*) FROM t,即使有主键索引,InnoDB 仍可能选 type=ALL,因为引擎直接遍历聚簇索引叶子节点更高效。

  • 真正要看的是 key 列:非 NULL 才说明用了哪个索引
  • rows 是预估扫描行数,不是实际返回行数;若远大于结果集,才值得怀疑索引有效性
  • 注意 Extra 中的 using filesortUsing temporary,比 type 更反映真实性能瓶颈

索引是否生效,得看具体 SQL 的谓词、排序、连接方式,以及数据分布。别只盯着 CREATE INDEX,先 EXPLAIN,再看 SHOW INDEX,最后验证 Handler<em>read</em>* 状态变量——很多慢查问题,其实出在索引根本没被选中,而不是索引建得不够多。

text=ZqhQzanResources