【MySQL-索引调优】06:最左匹配原则及优化

最左匹配原则

PS:现在orders表中存在idx_user_id(user_id),以及idx_user_status(user_id, status)两个索引

1-例子

运行:

sql 复制代码
EXPLAIN SELECT * FROM orders WHERE status = 1;

结果为:

id select_type table partitions type possible_keys key key_len ref rows filtered Extra
1 SIMPLE orders ALL 99520 10 Using where

分析:联合索引无法使用

2-分析

联合索引结构:

复制代码
(user_id, status)

索引的排序结构实际上是:

复制代码
user_id → status

可以理解为类似这样排序:

复制代码
user_id=1 status=0
user_id=1 status=1
user_id=1 status=2

user_id=2 status=0
user_id=2 status=1
user_id=2 status=2

所以索引查找必须:

复制代码
先确定 user_id
再确定 status

而你的 SQL:

复制代码
WHERE status = 1

缺少:

复制代码
user_id

数据库不知道从哪一段开始查找,因此:

复制代码
无法利用索引

于是优化器选择:

复制代码
全表扫描

执行计划变成:

复制代码
type: ALL

3-引出概念

联合索引:

复制代码
(user_id, status)

可使用的查询:

复制代码
WHERE user_id = 1
WHERE user_id = 1 AND status = 1

不可使用:

复制代码
WHERE status = 1

规则:

复制代码
必须从索引最左列开始使用

这就是:

复制代码
最左匹配原则

4-举例说明

4-1.联合索引的存储方式

假设你建了一个联合索引 (user_id, status, amount),它在底层的排序方式是

bash 复制代码
(user_id, status, amount)

也就是说,索引先按 user_id 排序,再在相同的 user_id 下按 status 排序,再在相同的 (user_id, status) 下按 amount 排序

4-2.最左匹配原则的含义

查询条件必须从联合索引的最左字段开始,才能利用索引

可以用到连续的前缀字段,但一旦中间断了,就不能继续往后匹配

举例说明:

  • WHERE user_id = 1 :能用索引,用到索引的第一列
  • WHERE user_id = 1 AND status = 1:能用索引,用到索引的前两列
  • WHERE user_id = 1 AND status = 1 AND amount = 100:能用索引,用到索引的三列
  • WHERE user_id = 1 AND amount = 100:这种能用到部分索引,只能吃到user_id这一列
  • WHERE status = 1:不能用索引,因为跳过了最左的 user_id
  • WHERE status = 1 AND amount = 100:也不能用索引,因为没有从最左的 user_id 开始

PS:这个和你写的sql条件的顺序不影响,你写的sql会被优化器会自动重排条件

sql 复制代码
SELECT * FROM orders WHERE status = 1 AND user_id = 1;
#status写在前,user_id写在后,依旧可以吃到(user_id, status, amount)索引

5-删除冗余索引

因为联合索引和最左匹配原则:

如果现在表里有一个(user_id)单列索引(user_id, status, amount)联合索引,那么这个(user_id)单列索引可以删除了,因为联合索引由user_id开头,完全能覆盖单列索引的功能

相关推荐
jiayou647 小时前
KingbaseES 表级与列级加密完全指南
数据库·后端
GBASE1 天前
G术时刻 |GBase 8s数据库事务并发控制之封锁技术介绍(下)
数据库
xiezhr2 天前
逛GitHub发现了一款免费的带AI功能的数据库管理工具
数据库·ai编程·dba
你听得到112 天前
用户说 App 卡,但说不清在哪?我把 Flutter 监控 SDK 升级成了链路观测工作台
前端·flutter·性能优化
唐青枫2 天前
MySQL JSON 实战详解:从存储、查询、更新到 JSON_TABLE 与索引
sql·mysql
吃糖的小孩2 天前
给 QQ AI 机器人设计“可控记忆”:会话摘要、手动长期记忆与角色卡边界
数据库
小满8782 天前
5.Mysql事务隔离级别与锁机制
mysql
笃行3503 天前
金仓数据库数据安全双防线:静态存储加密与传输加密实战
数据库
笃行3503 天前
金仓数据库物理备份实战:sys_rman 全流程演练与误覆盖抢救
数据库
笃行3503 天前
金仓数据库逻辑备份实战:从全库导出到 Schema 替换的完整闭环
数据库