【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开头,完全能覆盖单列索引的功能

相关推荐
2401_82422269几秒前
HTML怎么标注字数限制提示_HTML实时字数统计占位【详解】
jvm·数据库·python
y = xⁿ4 分钟前
MySQL八股知识合集
android·mysql·adb
稀饭过霍16 分钟前
数据类型【TINYINT、SMALLINT、INT、BIGINT、decimal(18,2)】表示意思
数据库
俺不要写代码20 分钟前
数据库:DML
数据库·oracle
这个DBA有点耶22 分钟前
两张百万级大表JOIN跑崩了?试试这3招
数据库·代码规范
IntMainJhy28 分钟前
「Flutter三方库sqflite的鸿蒙化适配与实战指南:从入门到踩坑的本地数据库开发全记录」
数据库·flutter·华为·信息可视化·数据库开发·harmonyos
counting money1 小时前
Spring框架基础(依赖注入-全注解形式)
java·数据库·spring
计算机安禾1 小时前
【Linux从入门到精通】第33篇:数据库MySQL/MariaDB安装与基础调优
linux·数据库·mysql
瀚高PG实验室1 小时前
ERROR: invalid input syntax for type integer: “a“
数据库·瀚高数据库
S1998_1997111609•X1 小时前
论next/js在打击省份及犯罪行为集团的系统分析[特殊字符]设计
网络·数据库·百度·ssh·开闭原则