MySQL索引最左匹配原则

news/2025/5/24 9:44:38

MySQL索引最左匹配原则

  • 一、案例一
    • 1、表与索引创建
    • 2、查询语句举例
    • 3、那么究竟用到了哪些索引呢?
    • 4、原因
  • 二、案例二
    • 1、表与索引创建
    • 2、查询语句举例
    • 3、原因
    • 4、最左匹配原则
  • 三、案例三
    • 1、表与索引创建
    • 2、查询语句举例
    • 3、原因以及补充资料


一、案例一

1、表与索引创建

DROP TABLE IF EXISTS med_computer_info;
CREATE TABLE med_computer_info (ID BIGINT NOT NULL AUTO_INCREMENT COMMENT '电脑id',HOST_NAME VARCHAR ( 64 ) NOT NULL COMMENT '电脑id地址',PORT VARCHAR ( 64 ) NOT NULL COMMENT '电脑端口',TYPE INT NOT NULL COMMENT '电脑类型',LAUNCH_DATE DATE NOT NULL COMMENT '电脑发布日期',MODIFIED TIMESTAMP NOT NULL COMMENT '记录修改时间',CREATED TIMESTAMP NOT NULL COMMENT '记录创建时间',PRIMARY KEY ( ID ),UNIQUE KEY INDEX_WORKER_NODE ( HOST_NAME, PORT, LAUNCH_DATE, TYPE ) 
) COMMENT = 'DB WorkerID Assigner for UID Generator',
ENGINE = INNODB;

2、查询语句举例

select * from med_computer_info where PORT=3306 and Type=1  ;
select * from med_computer_info where PORT=3307 and HOST_NAME='172.21.1.1'  ;
select * from med_computer_info where PORT=3308 AND HOST_NAME='172.21.1.2' AND TYPE=2  ;

3、那么究竟用到了哪些索引呢?

第一句:没用到索引,在聚集索引上从左至右依次扫描过滤;

第二句:用到了辅助索引INDEX_WORKER_NODE;

第三句:用到了辅助索引INDEX_WORKER_NODE,但是只有HOST_NAME和PORT条件是通过索引完成的,条件TYPE是依次扫描过滤完成的;

4、原因

因为辅助索引是B+树实现的,虽然可以指定多个列,但是每个列的比较优先级不一样,写在前面的优先比较。一旦出现遗漏,在B+树上就无法继续搜索了(通过补齐等措施解决的除外),因此是按照最左连续匹配来的。既然是在B+树上搜索,对于条件的比较自然是要求精确匹配(即"=“和"IN”)。不过顺序倒是可以颠倒,因为查询优化器重排序一下就好了。

第一句,由于缺少HOST_NAME,只能在聚集索引的叶节点上,从左至右的扫描,挨个比对;

第二句,可以直接在辅助索引上查找,被找到的子树的所有叶节点就是命中的记录;

第三句,缺少LAUNCH_DATE条件,所以只能先依据HOST_NAME和PROT在辅助索引上查找,找到的主键值作为候选记录,然后到聚集索引上读取对应记录,再比较TYPE条件是否满足。

二、案例二

1、表与索引创建

DROP TABLE IF EXISTS student;  
CREATE TABLE `student` (`id` int(11) NOT NULL AUTO_INCREMENT,`name` varchar(255) DEFAULT NULL,`cid` int(11) DEFAULT NULL,PRIMARY KEY (`id`),KEY `name_cid_INX` (`name`,`cid`),KEY `name_INX` (`name`)
) ENGINE=InnoDB AUTO_INCREMENT=8 DEFAULT CHARSET=utf8

2、查询语句举例

依据mysql索引最左匹配原则,两个索引都匹配上了,这个没有问题

EXPLAIN SELECT * FROM student WHERE name='小红';

在这里插入图片描述

判断条件是cid=1,而cid是(name,cid)复合索引的一部分,没有问题,可以进行index类型的索引扫描方式。explain显示结果使用到了索引,是index类型的方式。

EXPLAIN SELECT * FROM student WHERE cid=1;

在这里插入图片描述
cid字段的索引数据也是有序的情况下才能使用,什么时候才是有序的呢?观察可知,当然是在name字段是等值匹配的情况下,cid才是有序的。

EXPLAIN SELECT * FROM student WHERE cid=1 AND name='小红';

在这里插入图片描述

3、原因

index:这种类型表示是mysql会对整个该索引进行扫描。要想用到这种类型的索引,对这个索引并无特别要求,只要是索引,或者某个复合索引的一部分,mysql都可能会采用index类型的方式扫描。但是呢,缺点是效率不高,mysql会从索引中的第一个数据一个个的查找到最后一个数据,直到找到符合判断条件的某个索引。

ref:这种类型表示mysql会根据特定的算法快速查找到某个符合条件的索引,而不是会对索引中每一个数据都进行一 一的扫描判断,也就是所谓你平常理解的使用索引查询会更快的取出数据。而要想实现这种查找,索引却是有要求的,要实现这种能快速查找的算法,索引就要满足特定的数据结构。简单说,也就是索引字段的数据必须是有序的,才能实现这种类型的查找,才能利用到索引。

4、最左匹配原则

以该表的(name,cid)复合索引为例,它内部结构简单说就是下面这样排列的:
在这里插入图片描述

Mysql创建复合索引的规则是首先会对复合索引的最左边的,也就是第一个name字段的数据进行排序,在第一个字段的排序基础上,然后再对后面第二个的cid字段进行排序。其实就相当于实现了类似 order by name cid这样一种排序规则。

所以:第一个name字段是绝对有序的,而第二字段就是无序的了。所以通常情况下,直接使用第二个cid字段进行条件判断是用不到索引的,当然,可能会出现上面的使用index类型的索引。这就是所谓的mysql为什么要强调最左前缀原则的原因。

那么什么时候才能用到呢?
当然是cid字段的索引数据也是有序的情况下才能使用

三、案例三

1、表与索引创建

DROP TABLE IF EXISTS student;  
CREATE TABLE `student` (`id` int(11) NOT NULL AUTO_INCREMENT,`name` varchar(255) DEFAULT NULL,`cid` int(11) DEFAULT NULL,`home` VARCHAR(255) DEFAULT NULL,PRIMARY KEY (`id`),KEY `name_cid_INX` (`name`,`cid`),KEY `name_INX` (`name`)
) ENGINE=InnoDB AUTO_INCREMENT=8 DEFAULT CHARSET=utf8

2、查询语句举例

EXPLAIN SELECT * FROM student WHERE cid=1;

在这里插入图片描述

3、原因以及补充资料

比案例二多了一个home字段,由于辅助索引包含聚集索引,所以案例二的辅助索引为全部字段。

辅助索引包含了主键id用于回表操作,同时利用覆盖索引扫描可以更好的优化SQL。

索引可以加快数据的检索,减少IO开销,会占用磁盘空间,是一种用空间换时间的优化手段,同时更新操作会导致索引频繁的合并分裂,影响索引性能,在实际的业务开发中,如何根据业务场景去设计合适的索引是非常重要的

mysql查询优化器会判断纠正这条sql语句该以什么样的顺序执行效率最高,最后才生成真正的执行计划。所以,当然是我们能尽量的利用到索引时的查询顺序效率最高咯,所以mysql查询优化器会最终以这种顺序进行查询执行。

文章来源:https://blog.csdn.net/weixin_46146718/article/details/122862701
本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:https://dhexx.cn/news/show-18188.html

相关文章

Impala概述,特点,缺点,架构说明

1、概述 Impala是Cloudera公司推出,提供对HDFS、Hbase数据的高性能、低延迟的交互式SQL查询功能。基于Hive使用内存计算,兼顾数据仓库、具有实时、批处理、多并发等优点impala使用hive的元数据, 完全在内存中计算是CDH平台首选的PB级大数据实时查询分析引…

如何在页面中(html 等)设置IE浏览器的文档模式

最近在使用一个开源框架,发现该框架能跨浏览器和版本,但唯独在IE下的Quirks Model下表现不太好。一开始想着如何自己去完善在该模式下的问题,后来一想为何不找到一种方法,在代码里面设置好让页面在IE浏览器下面不呈现Quirks Model…

《深入理解计算机网络》读后小记 5、数据链路层

一、数据链路层基础 1、定义:数据链路层是真正用于数据传输的逻辑通道。 2、划分数据链路层的必要性: (1)、由于物理层传输介质的多样性和通信规程的各不相同,所以需要从逻辑意义上构建一条性能稳定、不受传输介质类型…

整数的除法产生一个浮点数

整数的除法产生一个浮点数一、整数的除法产生一个浮点数1、代码2、结果3、解析二、其他一、整数的除法产生一个浮点数 1、代码 Testvoid contextLoad17() {int a 1;int b 2;float f1 a / b;float f2 (float) a / b;float f3 (float) (a / b * 1.0);float f4 (float) (1.…

Linux系统中/dev/mtd与/dev/mtdblock的区别,即MTD字符设备和块设备的区别

转:http://www.crifan.com/linux_system_in__dev__mtd_and__dev__mtdblock_distinction_character_devices_and_block_devices_mtd_difference/ 1. /dev/mtdN 是Linux 中的MTD架构中,系统自己实现的mtd分区所对应的字符设备,其里面添加了一些…

Impala安装方式

Impala安装方式有两种 1 通过Cloudera Manager(CDH方式安装),推荐用这个,方便 2 手动安装,此方式级其复杂 下面通过Cloudera Manager的方式安装Impala 在主页中点击添加服务 选择Impala服务 进行角色分配 注意:最好将StateS…

车辆违章查询演示代码

车辆违章查询演示示例 using System; using System.Text; using System.Net; namespace weizhang { class Program { static void Main(string[] args) { string city "********";// 城市代码 * string hphm "********";// 号牌号码 完整7位 * strin…

Java8中的双冒号操作符

提示:文章写完后,目录可以自动生成,如何生成可参考右边的帮助文档 文章目录一、定义1、常用语法二、介绍1、静态方法语法2、类实例方法语法3、超类方法语法4、类构造器语法5、类实例方法语法三、案例一、定义 就是Java 8中的Lambda写法 双冒…

Impala的外部shell 操作命令

外部shell 命令:就是还没有进入Impala,运行 impala-shell,可添加的参数 选项描述-h, --help显示帮助信息-v or --version显示版本信息-i hostname, --impaladhostname指定连接运行 impalad 守护进程的主机。默认端口是 21000-q query, --que…

中值滤波和均值滤波C++代码

均值滤波和中值滤波代码 2008-11-24 16:07:36| 分类&#xff1a; 编程|举报|字号 订阅 //------------------均值滤波器 bool FilterAV(unsigned char *image,int height,int width) { int i,j; unsigned char *p(unsigned char*)malloc(height*width); for(i1;i<height-…