MySQL——最左前缀法则

最左前缀法则是 MySQL 联合索引中最核心的生效规则。

简单来说:如果你建立了一个包含多个列的索引,就像是在通讯录里先按"姓"排、再按"名"排一样,查询时必须从索引的最左列开始,且不能跳过中间的列。

一、 核心逻辑:像"找名字"一样思考

假设创建了一个联合索引:INDEX(col1, col2, col3)

在底层的 B+ 树中,数据的排序逻辑是:

先根据 col1 排序。

col1 相同的情况下,再根据 col2 排序。

col2 相同的情况下,再根据col3排序。

二、 三种典型的应用场景

为了方便理解,我们假设索引是 (name, age, position)

  1. 全值匹配(完全生效)
    如果你在 WHERE 子句里用到了所有索引列,顺序无所谓(MySQL 优化器会自动调整顺序),索引完全生效。

WHERE name='张三' AND age=20 AND position='Dev'

  1. 部分匹配(部分生效)
    必须从最左边开始,连续匹配。

WHERE name='张三' ✅(用到了 name 索引)

WHERE name='张三' AND age=20 ✅(用到了 nameage 索引)

  1. 违反规则(失效)
    中间断档:WHERE name='张三' AND position='Dev'

结果:只有 name 用到了索引。因为跳过了 ageposition 在索引树里是无序的,无法直接定位。

非最左开头:WHERE age=20 AND position='Dev'

结果:完全失效。因为没有 name,数据库根本不知道从 B+ 树的哪一个分支开始找,只能全表扫描。

三、 两个致命的"断点"

除了跳过列,还有两种情况会导致最左前缀法则的后续部分失效:

  1. 范围查询(后面失效)
    在联合索引中,如果中间某个列使用了范围查询(>、<、BETWEENLIKE 'Zhang%'),那么该列之后的索引列将失效。

WHERE name='张三' AND age > 20 AND position='Dev'

效果:name 有效,age 有效,但 position 失效。

  1. 模糊查询(前缀通配符)
    WHERE name LIKE '张%' ✅(生效)

WHERE name LIKE '%张' ❌(失效,因为开头不确定,破坏了最左原则)

四、 为什么会有这个法则?

这是由 B+ 树的排序特性决定的。

在多列索引的 B+ 树中,只有左边的列相等时,右边的列才有序。

如果你直接找 age=20 的人,在整个 B+ 树里,age=20 的记录可能散落在"张三"的分支里,也可能在"李四"的分支里。

数据库无法通过一次 B+ 树的搜索直接抓取所有 age=20 的人,只能挨个分支去找,这和全表扫描没区别。

总结口诀

带头大哥不能死,中间兄弟不能断,范围之后全完蛋。

相关推荐
倔强的石头_32 分钟前
《Kingbase护城河》——深度解密数据库行锁冲突与等待事件架构
数据库
IT策士41 分钟前
Redis 从入门到精通:性能调优与多语言客户端对比
数据库·redis·缓存
Bert.Cai1 小时前
Oracle INSTR函数详解
数据库·oracle
茉莉玫瑰花茶3 小时前
综合案例 - AI 智能租房助手 [ 5 ]
服务器·数据库·人工智能·python·ai
ywl4708120873 小时前
jwt生产token,简单版helloworld
java·数据库·spring
器灵科技4 小时前
AI视频工具实测:Seedance/可灵/HappyHorse谁最能打?
java·运维·数据库·人工智能·github
huangdong_4 小时前
京东商品图片视频批量下载与m3u8视频合并技术完整实现方案
大数据·前端·数据库
倒流时光三十年4 小时前
PostgreSQL CASE 条件表达式详解
数据库·postgresql
字节跳动数据平台5 小时前
营销视频进入工业化时代,火山引擎多模态数据湖如何助力多米实现内容生产提效 100+ 倍
数据库
健康平安的活着5 小时前
mysql中数据库脚本太大,通过脚本命令修改db名称
数据库·mysql