MySQL最左匹配原则深入分析

 更新时间:2022年11月01日 11:10:16   作者:流烟默  
首先回顾一下什么是最左匹配(也有称之为最左前缀)?顾名思义:最左优先,以最左边的为起点任何连续的索引都能匹配上。同时遇到范围查询(>、<、between、like)就会停止匹配

前言

接下来我们通过几种情况来描述最左匹配原则的使用。首先如下所示,为userName、phone以及userDate创建联合索引。

全列匹配

explain select * from user where userName ='admin' and phone ='13413413400' 
and userDate ='2000-04-29-12:53'

很明显,当按照索引中所有列进行精确匹配(这里精确匹配指“=”或“IN”匹配)时,索引可以被用到。这里有一点需要注意,理论上索引对顺序是敏感的,但是由于MySQL的查询优化器会自动调整where子句的条件顺序以使用适合的索引,例如我们将where中的条件顺序颠倒,其效果是一样的。

explain select * from user where userName ='admin'  
and userDate ='2000-04-29-12:53' and phone ='13413413400'

最左前缀匹配

比如我们where条件中只有userName:

explain select * from user where userName ='admin' 

当查询条件精确匹配索引的左边连续一个或几个列时,如userName,索引可以被用

到,但是只能用到一部分,即条件所组成的最左前缀。

上面的查询从分析结果看用到了 const 索引,key_len为152,说明只用到了索引的第一列前缀。

精确匹配

查询条件用到了索引中列的精确匹配,但是中间某个条件未提供

比如下面我们没有phone:

explain select * from user where userName ='admin' 
and userDate ='2000-04-29-12:53'

注意,上图Extra中值是Using index condition,说明MySQL正在使用覆盖索引,它只扫描索引的数据而不是按索引次序的每一行。它比按索引次序全表扫描的开销要少很多。

此时索引使用情况和情况二相同,因为phone未提供,所以查询只用到了索引的第一列,而后面的userDate虽然也在索引中,但是由于phone不存在而无法和左前缀连接,因此需要对结果进行扫描过滤userDate。

如果想让userDate也使用索引而不是where过滤,可以增加一个辅助索引<userName, userDate>,此时上面的查询会使用这个索引。

查询条件没有指定索引第一列

由于不是最左前缀,索引这样的查询显然用不到索引。

explain select * from user where userDate ='2000-04-29-12:53'

Using where:使用了用where子句来过滤结果集。这意味着MySQL服务器将在存储引擎检索行后再进行过滤。

匹配某列的前缀字符串

explain select * from user where userName ='admin' and phone like '134%' 
and  userDate ='2000-04-29-12:53'

此时可以用到索引,如果通配符%不出现在开头,则可以用到索引,但根据具体情况不同可能只会用其中一个前缀。

范围查询

explain select * from user where userName ='admin' and phone >'134' 
and  userDate ='2000-04-29-12:53'

范围列可以用到索引(必须是最左前缀),但是范围列后面的列无法用到索引。同时,索引最多用于一个范围列,因此如果查询条件中有两个范围列则无法全用到索引。

这里特别要说明MySQL一个有意思的地方,那就是仅用explain可能无法区分范围索引和多值匹配,因为在type中这两者都显示为range。同时,用了“between”并不意味着就是范围查询,例如下面的查询:

explain select * from user where userName ='admin' and phone 
between '13413413400' and '13513513500' and  userDate ='2000-04-29-12:53'

看起来是用了两个范围查询,但作用于phone上的“BETWEEN”实际上相当于“IN”,也就是说phone实际是多值精确匹配。可以看到这个查询用到了索引全部三个列。因此在MySQL中要谨慎地区分多值匹配和范围匹配,否则会对MySQL的行为产生困惑。

查询条件中含有函数或表达式

如果查询条件中含有函数或表达式,则MySQL不会为这列使用索引。

explain select * from user where userName ='admin' 
and left(phone ,6) and  userDate ='2000-04-29-12:53'

可以看到这个其实就是只用到了userName这一列的索引。

关于explain中各个关键字说明可以参考:认真学习传送门

到此这篇关于MySQL最左匹配原则深入分析的文章就介绍到这了,更多相关MySQL最左匹配内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • 详解MySQL8的新特性ROLE

    详解MySQL8的新特性ROLE

    这篇文章主要介绍了详解MySQL8的新特性ROLE的相关资料,帮助大家更好的理解和使用MySQL8,感兴趣的朋友可以了解下
    2020-11-11
  • MySQL的隐式类型转换整理总结

    MySQL的隐式类型转换整理总结

    隐式类型转换有无法命中索引的风险,在高并发、大数据量的情况下,命不中索引带来的后果非常严重。下面这篇文章主要给大家整理总结了关于MySQL的隐式转化,需要的朋友可以参考借鉴,下面来一起看看吧。
    2016-12-12
  • MySQL组合索引使用及说明

    MySQL组合索引使用及说明

    这篇文章主要介绍了MySQL组合索引使用及说明,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教
    2026-02-02
  • mysql本地安装详细步骤和验证方法(Windows、macOS、Ubuntu)

    mysql本地安装详细步骤和验证方法(Windows、macOS、Ubuntu)

    这篇文章主要介绍了mysql本地安装详细步骤和验证方法的相关资料,分别针对 Windows、macOS、Linux(Ubuntu)三大主流系统,文中通过代码介绍的非常详细,需要的朋友可以参考下
    2025-10-10
  • MySql中时间字段增加函数索引方式

    MySql中时间字段增加函数索引方式

    文章介绍了MySQL中对时间字段建索引的注意事项,指出在查询时对时间字段使用函数会导致索引失效,并建议通过函数索引或优化SQL避免该问题,同时提醒MySQL 8.0.13前版本不支持函数索引
    2025-10-10
  • MySQL数据库 Load Data 多种用法

    MySQL数据库 Load Data 多种用法

    这篇文章主要介绍MySQL数据库中 Load Data 数据的几种种用法,需要的朋友可以参考下面文章的解说
    2021-08-08
  • 使用SQL语句统计数据时sum和count函数中使用if判断条件的讲解

    使用SQL语句统计数据时sum和count函数中使用if判断条件的讲解

    今天小编就为大家分享一篇关于使用SQL语句统计数据时sum和count函数中使用if判断条件的讲解,小编觉得内容挺不错的,现在分享给大家,具有很好的参考价值,需要的朋友一起跟随小编来看看吧
    2019-02-02
  • mysql source 命令导入大的sql文件的方法

    mysql source 命令导入大的sql文件的方法

    本文将详细介绍mysql source 命令导入大的sql文件的方法;需要的朋友可以参考下
    2012-11-11
  • MySQL绿色版(zip解压版)的安装图文教程(mysql-5.6.22-win32.zip)

    MySQL绿色版(zip解压版)的安装图文教程(mysql-5.6.22-win32.zip)

    由于工作需要最近要开始研究MySQL了(看来学习都是逼出来的),本人对mysql没有研究,可以说一个小白。 下面就从安装开始吧,虽然网上关于这方面的东西很多,还是需要自己把操作过程写下来
    2016-06-06
  • 开启MySQL的binlog日志的方法步骤

    开启MySQL的binlog日志的方法步骤

    这篇文章主要介绍了开启MySQL的binlog日志的方法步骤,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2021-03-03

最新评论