MySQL联合索引设计中字段顺序、区分度与优化器行为示例详解

 更新时间:2025年11月13日 11:05:31   作者:代码怪兽大大作战  
索引设计是数据库优化中的关键环节,遵循三大黄金原则可显著提升查询性能,这篇文章主要介绍了MySQL联合索引设计中字段顺序、区分度与优化器行为的相关资料,文中通过示例代码介绍的非常详细,需要的朋友可以参考下,

在日常开发中,我们常常为查询加上联合索引,例如:

CREATE INDEX idx_unit_user ON t_user (unit_id, user_id);

但在很多项目里,还能看到这种写法:

CREATE INDEX idx_del_unit_user ON t_user (del_flag, unit_id, user_id);

del_flag 表示是否删除,只有 01 两种取值。

很多人认为“查询总有 del_flag=0 条件,索引当然要从它开始”,

但事实上,这样反而可能拖慢查询速度。

本文将系统讲清楚三个关键问题

  1. 联合索引字段顺序的重要性

  2. 区分度(Cardinality)对索引效率的影响

  3. MySQL 优化器是否会自动调整 WHERE 条件顺序

一、联合索引的匹配原理回顾

联合索引 (a, b, c) 的底层是一个 B+Tree

MySQL 检索时会按照索引定义的列顺序有序排列:

a → b → c

因此,它遵循 最左前缀原则(Leftmost Prefix Rule)

  • 可以命中 (a)(a,b)(a,b,c)

  • 但无法单独命中 (b)(c)(b,c)

这意味着:索引列的顺序决定了 MySQL 能否利用该索引。

二、区分度(Cardinality)是什么?

区分度是衡量字段“区分能力”的指标:

区分度 = 不同值数量 / 总记录数

可通过命令查看:

SHOW INDEX FROM your_table;

其中 Cardinality 表示索引中不同值的大致数量。

字段取值示例区分度是否适合放在索引前面
del_flag0/1极低
genderM/F极低
unit_id上千单位中高
user_id唯一极高

三、为什么低区分度字段放前面会拖慢查询?

假设你定义了:

CREATE INDEX idx_del_unit_user ON t_user (del_flag, unit_id, user_id);

del_flag 只有两种值(0、1)。

查询如下:

SELECT * FROM t_user WHERE del_flag = 0 AND unit_id = 1001;

索引的逻辑结构类似:

(del_flag=0) → [unit_id 排序 ...]

(del_flag=1) → [unit_id 排序 ...]

MySQL 实际上会扫描整个 (del_flag=0) 这半边索引树,

再在其中过滤出 unit_id=1001 的数据。

因为 del_flag 不能有效缩小数据范围,性能几乎无提升。

低区分度列放在前面时,索引分区极不均衡,效果有限。

四、优化设计:高区分度字段放前

如果查询模式是:

WHERE del_flag=0 AND unit_id=? AND user_id=?

更合理的索引应为:

CREATE INDEX idx_unit_user_del ON t_user (unit_id, user_id, del_flag);

执行顺序如下:

  1. MySQL 先根据 unit_id 定位;

  2. 再通过 user_id 精确匹配;

  3. 最后判断 del_flag=0

结果是:扫描范围更小,性能显著提升。

五、WHERE 条件顺序会影响吗?

很多人问:

“如果我写的 SQL 是 WHERE del_flag=0 AND unit_id=? AND user_id=?
那是不是应该把 unit_id 放前面?”

答案是:不用。

MySQL 优化器会自动调整逻辑顺序

MySQL 的优化器会:

  • 自动重排 WHERE 条件;

  • 根据各条件的“选择性”(区分度)判断最优的索引路径;

  • 但它不会改变索引的定义顺序

换句话说:

  • 写 SQL 的顺序不重要

  • 索引定义的顺序才重要

实测验证

索引:

CREATE INDEX idx_unit_user_del ON t_user (unit_id, user_id, del_flag);

两条 SQL:

EXPLAIN SELECT * FROM t_user 
WHERE del_flag=0 AND unit_id=1001 AND user_id=8888;

EXPLAIN SELECT * FROM t_user 
WHERE unit_id=1001 AND user_id=8888 AND del_flag=0;

结果完全一致:

key: idx_unit_user_del
key_len: ...
rows: 1
Extra: Using index condition

✅ 说明优化器自动识别了最优执行路径,
WHERE 条件顺序无关紧要。

但优化器不会“反转索引”

如果索引定义是:

CREATE INDEX idx_del_unit_user ON t_user (del_flag, unit_id, user_id);

那无论你写:

WHERE unit_id=1001 AND user_id=8888 AND del_flag=0;

还是反过来写,

优化器都无法跳过 del_flag 直接用 (unit_id, user_id)

只能从 del_flag=0 那个分支扫描,性能依然很差。

六、实战对比

查询索引是否命中说明
WHERE unit_id=? AND user_id=? AND del_flag=0(unit_id, user_id, del_flag)✅ 完整命中🚀 性能最优
WHERE del_flag=0 AND unit_id=? AND user_id=?(unit_id, user_id, del_flag)✅ 完整命中🚀 一样快
WHERE del_flag=0 AND unit_id=?(unit_id, user_id, del_flag)✅ 部分命中👍 仍快
WHERE del_flag=0(unit_id, user_id, del_flag)❌ 不命中最左前缀🐢 慢
WHERE unit_id=? AND user_id=?(del_flag, unit_id, user_id)❌ 无法跳过 del_flag🐢 慢

七、区分度与索引顺序的设计原则

原则说明
区分度优先高区分度列放在前(如 unit_id、user_id)
过滤性优先查询中最能减少扫描范围的条件放前
稳定性优先每次查询必带的条件(如 del_flag)放最后
低区分度列不单独建索引例如 0/1、状态、布尔值
用 EXPLAIN 验证执行计划理论与实际可能受统计信息影响

八、推荐实践模板

查询场景推荐索引
WHERE del_flag=0 AND unit_id=?(unit_id, del_flag)
WHERE del_flag=0 AND user_id=?(user_id, del_flag)
WHERE del_flag=0 AND unit_id=? AND user_id=?(unit_id, user_id, del_flag)

🚫 不推荐 (del_flag, unit_id, user_id)
✅ 推荐 (unit_id, user_id, del_flag)

九、总结

重点说明
✅ 索引顺序决定可用性最左前缀原则
✅ 区分度决定效率区分度高 → 放前面
✅ WHERE 条件顺序无关紧要优化器会自动重排
❌ 低区分度列放前浪费索引如 del_flag、status
✅ 正确索引能提升数十倍性能用 EXPLAIN 验证

💬 一句话总结:

MySQL 会自动优化 WHERE 条件顺序,但不会改变索引定义顺序。
因此,请始终把高区分度字段放在联合索引前列,
把低区分度的 del_flag、status 等放在最后。

到此这篇关于MySQL联合索引设计中字段顺序、区分度与优化器行为的文章就介绍到这了,更多相关MySQL联合索引字段顺序、区分度与优化器内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • 一台服务器部署两个独立的mysql数据库操作实例

    一台服务器部署两个独立的mysql数据库操作实例

    这篇文章主要给大家介绍了关于一台服务器部署两个独立的mysql数据库的相关资料,同一台服务器装两个数据库,可以通过虚拟化技术实现,文中通过代码介绍的非常详细,需要的朋友可以参考下
    2024-03-03
  • 解决maven打包缺少依赖class xxx for user defined function to_pinyin failed to load问题

    解决maven打包缺少依赖class xxx for user defined&

    在使用自定义函数时因依赖缺失导致报错,通过定位Maven打包配置发现缺少依赖包,解决方法是在pom.xml中添加maven-shade-plugin插件,实现依赖打包和类隔离,成功解决依赖加载问题
    2025-09-09
  • Mysql5.7中JSON操作函数使用说明

    Mysql5.7中JSON操作函数使用说明

    本文给大家分享的是在mysql5.7中操作json的函数的使用方法以及相关示例,非常的实用,有需要的小伙伴可以参考下
    2017-07-07
  • MySql事务原理介绍及特性

    MySql事务原理介绍及特性

    事务就是一组DML语句组成,这些语句在逻辑上存在相关性,这一组DML语句要么全部成功,要么全部失败,是一个整体。MySQL提供一种机制,保证我们达到这样的效果。事务还规定不同的客户端看到的数据是不相同的
    2022-09-09
  • MySQL分组查询、排序查询、分页查询以及执行顺序

    MySQL分组查询、排序查询、分页查询以及执行顺序

    MySQL数据查询是数据库操作中最常见的操作之一,它可以帮助我们从数据库中获取所需的数据,下面这篇文章主要给大家介绍了关于MySQL分组查询、排序查询、分页查询以及执行顺序的相关资料,需要的朋友可以参考下
    2024-02-02
  • centos7.3 安装mysql5.7.18的详细教程

    centos7.3 安装mysql5.7.18的详细教程

    这篇文章主要介绍了centos7.3 安装mysql5.7.18的详细教程,需要的朋友可以参考下
    2017-06-06
  • MySQL统计高频用户实现方法详解

    MySQL统计高频用户实现方法详解

    这篇文章主要介绍了MySQL统计高频用户实现的相关资料,文中通过示例代码讲解了如何用SQL实现数据清洗、分组聚合与排序,识别高频用户,用于活跃度分析和异常检测,需要的朋友可以参考下
    2025-05-05
  • 解析mysql中的auto_increment的问题

    解析mysql中的auto_increment的问题

    本篇文章是对mysql中的auto_increment的问题进行了详细的分析介绍,需要的朋友参考下
    2013-06-06
  • 快速实现MySQL的部署以及一机多实例部署

    快速实现MySQL的部署以及一机多实例部署

    这篇文章主要为大家详细介绍了快速实现MySQL的部署以及一机多实例部署的相关资料,感兴趣的小伙伴们可以参考一下
    2016-04-04
  • MySQL数据库MyISAM存储引擎转为Innodb的方法

    MySQL数据库MyISAM存储引擎转为Innodb的方法

    mysql数据库存储引擎为MyISAM的时候,在大访问量的情况下数据表有可能会出现被锁的情况,这就会导致用户连接网站时超时而返回502,此时就需要MySQL数据库MyISAM存储引擎转为Innodb,这篇文章主要介绍了MySQL数据库MyISAM存储引擎转为Innodb的方法,需要的朋友可以参考下
    2014-06-06

最新评论