MySQL 的EXPLAIN 计划 type 字段详细说明

在 MySQL 的 EXPLAIN 执行计划中,type 字段表示查询的访问类型,即 MySQL 在查询时使用的数据检索方法。不同的 type 值表示不同的查询效率,通常 system 是最优的,而 ALL 是最差的。以下是这些 type 值的详细说明及其排序:

复制代码
system > const > eq_ref > ref > range > index > ALL

1. system(性能最优)

  • 说明

    • 该表只有一行记录(如 system 表,或者只有一条记录的 MyISAM 表)。
    • 相当于 const,但只适用于单行数据的情况。
  • 示例

    sql 复制代码
    EXPLAIN SELECT * FROM singleton_table;
  • 特点

    • 由于表只有一行,直接读取该行即可,不涉及索引扫描或全表扫描。
    • 访问代价极低。

2. const(常量查询)

  • 说明

    • 适用于 唯一索引(PRIMARY KEY 或 UNIQUE) 查询,并且查询条件是常量。
    • MySQL 仅需查询一次,就能获取数据。
    • 适用于 WHERE 子句中的主键或唯一索引等查询。
  • 示例

    sql 复制代码
    EXPLAIN SELECT * FROM employees WHERE emp_id = 100;
  • 特点

    • 查询结果最多返回一行。
    • 查询优化器会在编译阶段就确定查询结果,速度极快。
    • 适用于主键或唯一索引的等值查询。

3. eq_ref(唯一索引等值匹配)

  • 说明

    • eq_ref 用于 主键(PRIMARY KEY)或唯一索引(UNIQUE KEY) 的等值查询,通常出现在多表 JOIN 操作中。
    • 对于每一行来自主表的数据,子表最多只会返回一条匹配记录。
  • 示例

    sql 复制代码
    EXPLAIN 
    SELECT e.emp_id, d.dept_name 
    FROM employees e 
    JOIN departments d ON e.dept_id = d.dept_id;
  • 特点

    • 每次查询只返回一行数据 (因为 eq_ref 必须基于 PRIMARY KEYUNIQUE 索引)。
    • 查询优化效果非常好。
    • 常见于 JOIN 语句中,被连接表的主键或唯一索引用于匹配主表的字段。

4. ref(普通索引等值匹配)

  • 说明

    • ref 表示查询使用了非唯一索引(普通索引、非唯一的外键索引)。
    • 适用于非唯一索引或 JOIN 操作中的索引匹配,可能会返回 多行 数据。
  • 示例

    sql 复制代码
    EXPLAIN SELECT * FROM employees WHERE department_id = 5;
  • 特点

    • 使用索引查找,但不是唯一匹配(可能匹配多行)。
    • 常用于 JOIN 操作,或者在 WHERE 子句中使用非唯一索引的查询。

5. range(范围查询)

  • 说明

    • range 表示基于索引范围扫描,可以使用索引高效查找数据。
    • 常见的范围查询有 BETWEEN><IN 等。
  • 示例

    sql 复制代码
    EXPLAIN SELECT * FROM employees WHERE salary BETWEEN 5000 AND 10000;
  • 特点

    • 适用于索引列上的范围查询。
    • 扫描索引的部分数据,比全表扫描(ALL)高效。
    • 适用于 >, <, BETWEEN, IN() 等操作符。

6. index(全索引扫描)

  • 说明

    • index 表示 全索引扫描 ,相当于 ALL(全表扫描),但扫描的是索引而非数据行。
    • 适用于索引覆盖查询(即不需要回表查询)。
  • 示例

    sql 复制代码
    EXPLAIN SELECT idx_column FROM large_table;
  • 特点

    • 比全表扫描(ALL)稍快,因为索引通常比整个表小。
    • 不会使用 WHERE 过滤条件,会遍历整个索引。

7. ALL(全表扫描,性能最差)

  • 说明

    • ALL 表示 全表扫描,意味着 MySQL 需要遍历整个表的所有数据行。
    • 适用于无索引的查询,或者查询无法利用索引。
  • 示例

    sql 复制代码
    EXPLAIN SELECT * FROM employees WHERE name = 'John Doe';
  • 特点

    • 性能最差,因为需要扫描整张表。
    • MySQL 可能会自动使用 Using filesortUsing temporary 进行排序。
    • 如果 ALL 出现在大表查询中,通常需要优化索引

总结与排序

访问类型 适用情况 典型场景 速度
system 只有一行数据的表 读取单行数据 最快
const 主键或唯一索引等值查询 WHERE id = 1 极快
eq_ref 主键或唯一索引 JOIN 查询 JOIN ON primary_key 非常快
ref 非唯一索引等值查询 WHERE department_id = 5
range 范围查询 WHERE age BETWEEN 18 AND 30 中等
index 全索引扫描 SELECT idx_column FROM table 较慢
ALL 无索引查询或全表扫描 SELECT * FROM large_table 最慢

优化建议

  1. 避免 ALL(全表扫描)

    • 创建索引,提高查询效率。
    • 使用 EXPLAIN 分析执行计划,确认索引是否被使用。
  2. 优化 range(范围扫描)

    • 适当调整索引,避免 MySQL 过度扫描数据范围。
    • 使用 BETWEENIN() 查询时,确保索引列的数据分布合理。
  3. 提升 refeq_ref 查询

    • 使用外键关系,并确保 JOIN 语句涉及索引列。
    • 采用 覆盖索引,减少回表查询的次数。
  4. 优化 index(全索引扫描)

    • 避免 SELECT *,只查询必要字段,减少数据量。
    • 确保查询能利用索引,避免额外的 Using filesortUsing temporary

通过这些优化方法,可以减少数据库的负载,提高查询速度。

相关推荐
fen_fen2 小时前
用户信息表建表及批量插入 100 条数据(MySQL/Oracle)
数据库·mysql·oracle
马克Markorg8 小时前
常见的向量数据库和具有向量数据库能力的数据库
数据库
百锦再8 小时前
React编程高级主题:测试代码
android·前端·javascript·react.js·前端框架·reactjs
2501_9160088910 小时前
全面介绍Fiddler、Wireshark、HttpWatch、SmartSniff和firebug抓包工具功能与使用
android·ios·小程序·https·uni-app·iphone·webview
Coder_Boy_10 小时前
技术让开发更轻松的底层矛盾
java·大数据·数据库·人工智能·深度学习
玉梅小洋10 小时前
Windows 10 Android 构建配置指南
android·windows
helloworldandy10 小时前
使用Pandas进行数据分析:从数据清洗到可视化
jvm·数据库·python
Libraeking12 小时前
视觉篇:Canvas 自定义绘图与高级动画的华丽圆舞曲
android·经验分享·android jetpack
数据知道12 小时前
PostgreSQL 故障排查:如何找出数据库中最耗时的 SQL 语句
数据库·sql·postgresql
qq_124987075312 小时前
基于SSM的动物保护系统的设计与实现(源码+论文+部署+安装)
java·数据库·spring boot·毕业设计·ssm·计算机毕业设计