什么是组合索引?

  由多个字段组成的索引叫组合索引。

问题:在哪些场景中,组合索引会失效?

场景

数据表:job_status_trace_log【说明:id是主键】
数据量:35w
创建索引:ALTER table job_status_trace_log add INDEX creation_time_index(creation_time,job_name,source)

注意:创建表的时候,字段要超过组合索引字段的个数,因为它会命中覆盖索引,导致跟我下面测试的数据不一致

例子:组合索引(a,b,c)都有哪些排列组合

  • a,b,c
  • a,c,b
  • c,a,b
  • c,b,a
  • b,c,a
  • a,b
  • b,a
  • a,c
  • c,a
  • b,c
  • a
  • b
  • c
  • …就列举那么多已经够了

先看看总结,再看验证过程,可能会更棒

总结:

1、组合索引字段无论顺序如何改变都会用到索引,前提是所有字段都在where条件上
2、如果想要使用一个或者两个字段在where条件上,必须有组合索引里的第一个字段,但是与顺序无关,例如a,c或c,a,这种场景是可以命中索引的。但是,b,c或c,b这种是不会命中索引的。
3、如果组合索引存在范围查询,则组合索引可能会命中索引,这个跟B+Tree的叶子节点中存储的数据是否在当前的叶子节点中,即InnoDB存储引擎的最小存储单元——页,InnoDB页的大小默认是16k,可以通过参数查看页的默认大小:show global status like ‘innodb_page_size’;如果想要修改InnoDB页的大小,需要通过修改mysql源码才可以修改,找到源码文件(storage/innobase/include/univ.i),找到参数:UNIV_PAGE_SIZE,该参数必须是2的n次方,例如4k、8k、16k、32k、64k等等。
4、order by 只能使用a,才能用到索引

排列组合一:a,b,c,恭喜成功命中索引
mysql> explain select * from job_status_trace_log where creation_time = '2020-01-01 00:00:00' and job_name ='member.channelRouteTask' and source='LITE_EXECUTOR'\G
*************************** 1. 行 ***************************
id           : 1
select_type  : SIMPLE
table        : job_status_trace_log
partitions   : NULL
type         : ref
possible_keys: creation_time_index
key          : creation_time_index
key_len      : 609
ref          : const,const,const
rows         : 1
filtered     : 100.00
Extra        : NULL
1 行于数据集 (0.02 秒)
排列组合二:a,c,b,恭喜成功命中索引
mysql> explain select * from job_status_trace_log where creation_time = '2020-01-01 00:00:00' and source='LITE_EXECUTOR' and job_name ='member.channelRouteTask'\G
*************************** 1. 行 ***************************
id           : 1
select_type  : SIMPLE
table        : job_status_trace_log
partitions   : NULL
type         : ref
possible_keys: creation_time_index
key          : creation_time_index
key_len      : 609
ref          : const,const,const
rows         : 1
filtered     : 100.00
Extra        : NULL
1 行于数据集 (0.03 秒)
排列组合三:c,a,b,恭喜成功命中索引
mysql> explain select * from job_status_trace_log where source='LITE_EXECUTOR' and creation_time = '2020-01-01 00:00:00' and job_name ='member.channelRouteTask'\G
*************************** 1. 行 ***************************
id           : 1
select_type  : SIMPLE
table        : job_status_trace_log
partitions   : NULL
type         : ref
possible_keys: creation_time_index
key          : creation_time_index
key_len      : 609
ref          : const,const,const
rows         : 1
filtered     : 100.00
Extra        : NULL
1 行于数据集 (0.03 秒)
排列组合四:c,b,a,恭喜成功命中索引
mysql> explain select * from job_status_trace_log where source='LITE_EXECUTOR' and job_name ='member.channelRouteTask' and creation_time = '2020-01-01 00:00:00'\G
*************************** 1. 行 ***************************
id           : 1
select_type  : SIMPLE
table        : job_status_trace_log
partitions   : NULL
type         : ref
possible_keys: creation_time_index
key          : creation_time_index
key_len      : 609
ref          : const,const,const
rows         : 1
filtered     : 100.00
Extra        : NULL
1 行于数据集 (0.07 秒)
排列组合五:b,c,a,恭喜成功命中索引
mysql> explain select * from job_status_trace_log where job_name ='member.channelRouteTask' and source='LITE_EXECUTOR' and creation_time = '2020-01-01 00:00:00'\G
*************************** 1. 行 ***************************
id           : 1
select_type  : SIMPLE
table        : job_status_trace_log
partitions   : NULL
type         : ref
possible_keys: creation_time_index
key          : creation_time_index
key_len      : 609
ref          : const,const,const
rows         : 1
filtered     : 100.00
Extra        : NULL
1 行于数据集 (0.02 秒)
排列组合六:a,b,恭喜成功命中索引
mysql> explain select * from job_status_trace_log where creation_time = '2020-01-01 00:00:00' and job_name ='member.channelRouteTask'\G
*************************** 1. 行 ***************************
id           : 1
select_type  : SIMPLE
table        : job_status_trace_log
partitions   : NULL
type         : ref
possible_keys: creation_time_index
key          : creation_time_index
key_len      : 407
ref          : const,const
rows         : 2
filtered     : 100.00
Extra        : NULL
1 行于数据集 (0.04 秒)
排列组合七:b,a,恭喜成功命中索引
mysql> explain select * from job_status_trace_log where job_name ='member.channelRouteTask' and creation_time = '2020-01-01 00:00:00'\G
*************************** 1. 行 ***************************
id           : 1
select_type  : SIMPLE
table        : job_status_trace_log
partitions   : NULL
type         : ref
possible_keys: creation_time_index
key          : creation_time_index
key_len      : 407
ref          : const,const
rows         : 2
filtered     : 100.00
Extra        : NULL
1 行于数据集 (0.03 秒)
排列组合八:a,c,恭喜成功命中索引
mysql> explain select * from job_status_trace_log where creation_time = '2020-01-01 00:00:00' and source='11'\G
*************************** 1. 行 ***************************
id           : 1
select_type  : SIMPLE
table        : job_status_trace_log
partitions   : NULL
type         : ref
possible_keys: creation_time_index
key          : creation_time_index
key_len      : 5
ref          : const
rows         : 17
filtered     : 10.00
Extra        : Using index condition
1 行于数据集 (0.04 秒)
排列组合九:c,a,恭喜成功命中索引
mysql> explain select * from job_status_trace_log where source='11' and creation_time = '2020-01-01 00:00:00'\G
*************************** 1. 行 ***************************
id           : 1
select_type  : SIMPLE
table        : job_status_trace_log
partitions   : NULL
type         : ref
possible_keys: creation_time_index
key          : creation_time_index
key_len      : 5
ref          : const
rows         : 17
filtered     : 10.00
Extra        : Using index condition
1 行于数据集 (0.04 秒)
排列组合十:b,c,很遗憾全表扫描了
mysql> explain select * from job_status_trace_log where job_name ='member.channelRouteTask' and source='LITE_EXECUTOR'\G
*************************** 1. 行 ***************************
id           : 1
select_type  : SIMPLE
table        : job_status_trace_log
partitions   : NULL
type         : ALL
possible_keys: NULL
key          : NULL
key_len      : NULL
ref          : NULL
rows         : 4316676
filtered     : 1.00
Extra        : Using where
1 行于数据集 (0.03 秒)
排列组合十一:a,恭喜成功命中索引
mysql> explain select * from job_status_trace_log where creation_time = '2020-01-01 00:00:00'\G
*************************** 1. 行 ***************************
id           : 1
select_type  : SIMPLE
table        : job_status_trace_log
partitions   : NULL
type         : ref
possible_keys: creation_time_index
key          : creation_time_index
key_len      : 5
ref          : const
rows         : 17
filtered     : 100.00
Extra        : NULL
1 行于数据集 (0.02 秒)
排列组合十二:b,很遗憾全表扫描了
mysql> explain select * from job_status_trace_log where job_name ='member.channelRouteTask'\G
*************************** 1. 行 ***************************
id           : 1
select_type  : SIMPLE
table        : job_status_trace_log
partitions   : NULL
type         : ALL
possible_keys: NULL
key          : NULL
key_len      : NULL
ref          : NULL
rows         : 4316693
filtered     : 10.00
Extra        : Using where
1 行于数据集 (0.04 秒)
排列组合十三:c,很遗憾全表扫描了
mysql> explain select * from job_status_trace_log where source='LITE_EXECUTOR'\G
*************************** 1. 行 ***************************
id           : 1
select_type  : SIMPLE
table        : job_status_trace_log
partitions   : NULL
type         : ALL
possible_keys: NULL
key          : NULL
key_len      : NULL
ref          : NULL
rows         : 4316702
filtered     : 10.00
Extra        : Using where
1 行于数据集 (0.06 秒)
排列组合十四:a>0,b=1,c=1,很遗憾全表扫描了
mysql> explain select * from job_status_trace_log where creation_time > '2020-01-01 00:00:00' and job_name ='member.channelRouteTask' and source='LITE_EXECUTOR'\G
*************************** 1. 行 ***************************
id           : 1
select_type  : SIMPLE
table        : job_status_trace_log
partitions   : NULL
type         : ALL
possible_keys: creation_time_index
key          : NULL
key_len      : NULL
ref          : NULL
rows         : 4316703
filtered     : 0.38
Extra        : Using where
1 行于数据集 (0.04 秒)
排列组合十五:a=1,b>0,c=1,很遗憾全表扫描了
mysql> explain select * from job_status_trace_log where  job_name ='member.channelRouteTask' and creation_time > '2020-01-01 00:00:00' and source='LITE_EXECUTOR'\G
*************************** 1. 行 ***************************
id           : 1
select_type  : SIMPLE
table        : job_status_trace_log
partitions   : NULL
type         : ALL
possible_keys: creation_time_index
key          : NULL
key_len      : NULL
ref          : NULL
rows         : 4316727
filtered     : 0.38
Extra        : Using where
1 行于数据集 (0.02 秒)
排列组合十六:a=1,b=1,c>0,很遗憾全表扫描了
mysql> explain select * from job_status_trace_log where  job_name ='member.channelRouteTask'  and source='LITE_EXECUTOR' and creation_time > '2020-01-01 00:00:00'\G
*************************** 1. 行 ***************************
id           : 1
select_type  : SIMPLE
table        : job_status_trace_log
partitions   : NULL
type         : ALL
possible_keys: creation_time_index
key          : NULL
key_len      : NULL
ref          : NULL
rows         : 4316730
filtered     : 0.38
Extra        : Using where
1 行于数据集 (0.06 秒)
排列组合十七:a>0,很遗憾全表扫描了
mysql> explain select * from job_status_trace_log where creation_time > '2020-01-01 00:00:00'\G
*************************** 1. 行 ***************************
id           : 1
select_type  : SIMPLE
table        : job_status_trace_log
partitions   : NULL
type         : ALL
possible_keys: creation_time_index
key          : NULL
key_len      : NULL
ref          : NULL
rows         : 4316730
filtered     : 38.09
Extra        : Using where
1 行于数据集 (0.03 秒)
总结:
  • 1、组合索引字段无论顺序如何改变都会用到索引,前提是所有字段都在where条件上。
  • 2、如果想要使用一个或者两个字段在where条件上,必须有组合索引里的第一个字段,但是与顺序无关,例如a,c或c,a,这种场景是可以命中索引的。但是,b,c或c,b这种是不会命中索引的。
  • 3、如果组合索引存在范围查询,则组合索引可能会命中索引,这个跟B+Tree的叶子节点中存储的数据是否在当前的叶子节点中,即InnoDB存储引擎的最小存储单元——页,InnoDB页的大小默认是16k,可以通过参数查看页的默认大小:show global status like ‘innodb_page_size’;如果想要修改InnoDB页的大小,需要通过修改mysql源码才可以修改,找到源码文件(storage/innobase/include/univ.i),找到参数:UNIV_PAGE_SIZE,该参数必须是2的n次方,例如4k、8k、16k、32k、64k等等。
  • 4、order by 只能使用a,才能用到索引。

欢迎关注我的微信公众号,里面有很多干货,各种面试题

什么是组合索引?在哪些场景中,组合索引会失效?相关推荐

  1. mysql索引ab和ba_Mysql中的索引

    索引的常见模型有哈希表.有序数组和搜索树. 哈希表:一种以 KV 存储数据的结构,只适合等值查询,不适合范围查询. 有序数组:只适用于静态存储引擎,涉及到插入的时候比较麻烦.可以参考 Java 中的 ...

  2. mysql5.6 函数索引_聊聊MySQL中的索引

    关于MySQL中的索引使用 索引是数据库优化中最常用也是最重要的手段之一,通过索引通常可以帮助用户解决大多数的SQL性能问题. 索引的存储分类: 1.B-Tree索引:最常见的索引类型,大部分引擎都支 ...

  3. mysql复合索引的应用场景_mysql复合索引(联合索引)的使用场景

    什么是联合索引? 联合索引又叫复合索引.对于复合索引:Mysql从左到右的使用索引中的字段,一个查询可以只使用索引中的一部份,但只能是最左侧部分.例如索引是key index (a,b,c). 可以支 ...

  4. oracle 修改索引的名称,Oracle中查询索引名称,批量修改索引名称语句

    在Oralce数据库数据优化过程中,对源数据表处理,原则上是做更名备份,作为被查或回退使用,所以,有修改数据表名后重新建表的操作,这样,往往也需要修改索引.主键.外键名称,方便重建,为了方便.快速生成 ...

  5. mysql中创建唯一索引的关键字_mysql中唯一索引的关键字是什么

    mysql中唯一索引的关键字是unique index.创建唯一索引可以避免数据出现重复.唯一索引可以有多个,但索引列的值必须唯一,索引列的值允许有空值.创建唯一索引可以使用关键字UNIQUE随表一同 ...

  6. mysql唯一索引的关键字_mysql中唯一索引的关键字是什么

    mysql中唯一索引的关键字是unique index.创建唯一索引可以避免数据出现重复.唯一索引可以有多个,但索引列的值必须唯一,索引列的值允许有空值.创建唯一索引可以使用关键字UNIQUE随表一同 ...

  7. oracle 视图能建索引吗,Oracle视图中建立索引注意事项.doc

    Oracle视图中建立索引的注意事项 在视图上创建索引需要三个条件:一.视图必须绑定到架构.要做到这点,在?CREATE?VIEW?语句中,必须加上?WITH?SCHEMABINDING,如果是使用企 ...

  8. oracle 删掉索引,如何清除Oracle中无用索引

    DML性能低下,其中最严重的原因之一是无用索引的存在.所有SQL的插入,更新和删除操作在它们需要在每一行数据被改变时修改大量索引的时候会变得更慢. 许多Oracle 管理人员只要看见在一个SQL 查询 ...

  9. mysql索引添加缓慢_mysql 中 创建索引很慢,怎么解决

    引用 如题,我现在 有一张表...里面的数据大概就是 800w 条左右,当然以后也可能会更多,这个表会频繁的更新! 我现在的处理是:每次更新 都会先truncate 这张表(因为里面的数据 已经不需要 ...

  10. mysql普通索引自增_mysql中联合索引中的自增列的增长策略

    <深入理解MySQL>中一段介绍MyISAM存储引擎中自动增长列的示例,如下 1 mysql>create table autoincre_demo2 -> (d1 small ...

最新文章

  1. leetcode算法题--扑克牌中的顺子
  2. 改变按钮在iPhone下的默认风格
  3. 大数据学习——sparkRDD
  4. 雅马哈发电机换机油教程_奥迪老A4B7 EA113 1.8T发动机严重烧机油大修彻底解决
  5. Content-Type简要说明
  6. centos tomcat 安装
  7. 浅析 Linux 初始化系统(系统服务管理和控制程序/init system) -- UpStart
  8. iOS-Runtime-Headers
  9. 经常见到的监控摄像机的镜头
  10. 图片 滚动切换效果(五) 高级篇
  11. 面向对象的三个特点:封装、继承、多态
  12. 使用Visual Studio进行单元测试-Part4
  13. android5.0百度地图,百度地图安卓版5.0.0 官方版
  14. fastboot刷机操作
  15. 教大家一个快速批量去水印下载快手视频、图集的方法技巧
  16. 商用密码产品认证-数字证书认证系统
  17. Cadence PSpice 仿真3: 电容器充放电瞬态仿真图文教程
  18. 软件工程基础知识-软件质量
  19. 台式计算机配置作业,大学计算机上机课作业.doc
  20. centos7上运行 ultravnc repeater

热门文章

  1. imail 删除历史邮件命令
  2. javascript trim函数在IE下不能用
  3. 2023暑期实习历程总结
  4. 1144: 零起点学算法51——数组中删数(C语言)
  5. JdbcTemplate查询与批量更新
  6. 快递查询教程,批量查询物流,正在途中、已签收的单号怎么查询
  7. 做一个网站需要多少钱?有哪些费用组成?
  8. 利用 UMA 使硬件加速器可直接用于 TVM
  9. Java程序员上班“划水”向阿里猛投简历,两次被刷后,最终敲定“菜鸟网络”的岗位,定级P6.
  10. 36.使用FCL中的委托声明