MySQL 索引(Index)系统讲解

MySQL 中的索引 本质上是一种用于加速数据检索的数据结构 。理解索引,核心抓住三点:是什么、怎么用、什么时候会失效


一、索引是什么 & 为什么快

索引类似书的目录

  • 不用从头到尾扫描(全表扫描)
  • 通过有序结构,快速定位数据行

在 InnoDB 中,索引的底层实现是 B+Tree

为什么是 B+Tree?

  • 树高度低(通常 2~4 层)
  • 叶子节点有序 + 双向链表
  • 范围查询效率极高
  • 适合磁盘 IO(一次 IO 读一个页)

二、MySQL 索引的分类(按逻辑)

1. 主键索引(PRIMARY KEY)

  • 表中唯一
  • 不允许 NULL
  • InnoDB 中是聚簇索引
sql 复制代码
PRIMARY KEY (id)

数据本身存放在主键 B+Tree 的叶子节点中


2. 普通索引(INDEX)

  • 仅用于加速查询
  • 可重复、可为 NULL
sql 复制代码
CREATE INDEX idx_name ON user(name);

3. 唯一索引(UNIQUE)

  • 值必须唯一(NULL 除外)
sql 复制代码
CREATE UNIQUE INDEX uk_email ON user(email);

4. 联合索引(复合索引)

  • 多个字段组成
  • 最左前缀原则
sql 复制代码
CREATE INDEX idx_abc ON t(a, b, c);

✅ 能用索引的情况:

  • a
  • a, b
  • a, b, c

❌ 不能直接用:

  • b
  • b, c

5. 全文索引(FULLTEXT)

  • 用于文本搜索(MATCH ... AGAINST
  • 常用于文章、评论
sql 复制代码
FULLTEXT(title, content)

三、聚簇索引 vs 非聚簇索引(重点)

🔹 聚簇索引(Clustered Index)

  • 主键索引
  • 叶子节点 = 整行数据
  • 一张表只能有一个

🔹 二级索引(辅助索引)

  • 叶子节点存的是:主键值

  • 查询流程:

    1. 先走二级索引
    2. 拿到主键
    3. 再回表查数据(回表)

覆盖索引可以避免回表


四、覆盖索引(非常重要)

sql 复制代码
SELECT name FROM user WHERE name = 'Tom';
  • name 字段有索引
  • 查询字段 全在索引中
  • Extra: Using index

性能最好


五、索引什么时候会失效(高频面试)

以下情况索引可能失效:

  1. 最左前缀被破坏
sql 复制代码
WHERE b = 2 AND c = 3
  1. 对索引列使用函数
sql 复制代码
WHERE DATE(create_time) = '2026-01-01'
  1. 隐式类型转换
sql 复制代码
WHERE phone = 13800138000  -- phone 是 varchar
  1. LIKE '%xxx'
sql 复制代码
LIKE '%abc'
  1. OR 条件一边没索引
sql 复制代码
WHERE a = 1 OR b = 2

六、如何判断索引是否生效

1. EXPLAIN

sql 复制代码
EXPLAIN SELECT * FROM user WHERE email = 'a@b.com';

重点字段:

  • type(性能从好到差)
    const > ref > range > index > ALL

  • key:使用的索引

  • rows:扫描行数

  • Extra

    • Using index(覆盖索引)
    • Using where
    • Using filesort(危险)

七、索引设计原则(实战)

该建索引

  • 高频 WHERE
  • 高频 JOIN
  • 高频 ORDER BY
  • 区分度高的字段

不建议建索引

  • 更新极其频繁的字段
  • 区分度很低(如性别)
  • 小表(数据量 < 1w)

联合索引优于多个单列索引


八、面试一句话总结

MySQL InnoDB 使用 B+Tree 作为索引结构,主键索引是聚簇索引,数据行存储在主键索引的叶子节点中,普通索引需要回表;合理使用联合索引、覆盖索引,并遵循最左前缀原则,才能发挥索引的最大性能。


相关推荐
2301_795099741 小时前
golang如何在Gin中自定义验证器_golang Gin自定义验证器实现方法
jvm·数据库·python
2301_766283441 小时前
如何在MongoDB GridFS中进行按文件大小(length)范围的查询
jvm·数据库·python
布吉岛的石头1 小时前
分库分表实战:Sharding-JDBC 快速落地
分布式·mysql
冬天vs不冷2 小时前
面试必知必会(13):MySQL锁机制
mysql·面试·职场和发展
冬天vs不冷2 小时前
面试必知必会(14):MySQL执行计划与SQL优化
sql·mysql·面试
萧曵 丶2 小时前
MySQL 高频面试题(由浅到深 完整版,面试必背)
数据库·mysql·面试
czlczl200209252 小时前
MySQL 执行引擎:排序与临时表机制深度解析
数据库·mysql
lifewange3 小时前
DBeaver如何安装
数据库
m0_631529823 小时前
CSS如何利用CSS变量进行渐变色管理_提升渐变配置的灵活性
jvm·数据库·python
2301_818008443 小时前
数据库模型设计实战:如何正向工程从模型建表_规范化项目开发流程
jvm·数据库·python