什么是索引下推?

什么是索引下推

索引下推(Index Condition Pushdown,简称ICP),是MySQL5.6版本的新特性,用于优化数据查询。

不使用索引条件下推优化时存储引擎通过索引检索到数据,然后返回给MySQL服务器,服务器然后判断数据是否符合条件。

当使用索引条件下推优化时,如果存在某些被索引的列的判断条件时,MySQL服务器将这一部分判断条件传递给存储引擎,然后由存储引擎通过判断索引是否符合MySQL服务器传递的条件,只有当索引符合条件时才会将数据检索出来返回给MySQL服务器。

索引条件下推优化可以减少存储引擎查询基础表的次数,也可以减少MySQL服务器从存储引擎接收数据的次数。

换句话说:索引下推能减少回表查询次数,提高查询效率。

索引下推优化的原理

我们先简单了解一下MySQL大概的架构:

 

MySQL从上至下分为以下几层:

  • MySQL服务层:包括NoSQL和SQL接口、查询解析器、优化器、缓存和Buffer等组件。
  • 存储引擎层:各种插件式的表格存储引擎,实现事务、索引等各种存储引擎相关的特性。
  • 文件系统层: 读写物理文件。

MySQL服务层负责SQL语法解析、生成执行计划等,并调用存储引擎层去执行数据的存储和检索。
索引下推的下推其实就是指将部分上层(服务层)负责的事情,交给了下层(引擎层)去处理。
我们来具体看一下,在没有使用ICP的情况下,MySQL的查询:

  • 获取下一行,首先读取索引信息,然后根据索引将整行数据读取出来。
  • 然后通过where条件判断当前数据是否符合条件,符合返回数据。

使用ICP的情况下,查询过程:

  • 获取下一行的索引信息。
  • 检查索引中存储的列信息是否符合索引条件,如果符合将整行数据读取出来,如果不符合跳过读取下一行。
  • 用剩余的判断条件,判断此行数据是否符合要求,符合要求返回数据。

索引下推适用条件

  • 需要整表扫描的情况。比如:range, ref, eq_ref, ref_or_null 。
  • 适用于InnoDB 引擎和 MyISAM 引擎的查询。(5.6版本不适用分区表查询,5.7版本后可以用于分区表查询)。
  • 对于InnDB引擎只适用于二级索引,因为InnDB的聚簇索引会将整行数据读到InnDB的缓冲区,这样一来索引条件下推的主要目的减少IO次数就失去了意义。因为数据已经在内存中了,不再需要去读取了。
  • 引用子查询的条件不能下推。
  • 调用存储过程的条件不能下推,存储引擎无法调用位于MySQL服务器中的存储过程。
  • 触发条件不能下推。

EXPLAN分析当使用explan进行分析时,如果使用了索引条件下推,Extra会显示Using index condition。并不是Using index。

因为并不能确定利用索引条件下推查询出的数据就是符合要求的数据,还需要通过其他的查询条件来判断。

索引下推的具体实践

理论比较抽象,我们来上一个实践。使用一张用户表tuser,表里创建联合索引(name, age)。


如果现在有一个需求:检索出表中名字第一个字是张,而且年龄是10岁的所有用户。那么,SQL语句是这么写的。

select * from tuser where name like '张%' and age=10;

假如你了解索引最左匹配原则,那么就知道这个语句在搜索索引树的时候,只能用 张,找到的第一个满足条件的记录id为1。

 


那接下来的步骤是什么呢?

没有使用ICP

在MySQL 5.6之前,存储引擎根据通过联合索引找到name likelike ‘张%’ 的主键id(1、4),逐一进行回表扫描,去聚簇索引找到完整的行记录,server层再对数据根据age=10进行筛选。

我们看一下示意图:


可以看到需要回表两次,把我们联合索引的另一个字段age浪费了。

使用ICP

而MySQL 5.6 以后, 存储引擎根据(name,age)联合索引,找到,由于联合索引中包含列,所以存储引擎直接再联合索引里按照age=10过滤。按照过滤后的数据再一一进行回表扫描。

我们看一下示意图:

 


可以看到只回表了一次。

除此之外我们还可以看一下执行计划,看到Extra一列里 Using index condition,这就是用到了索引下推。

+----+-------------+-------+------------+-------+---------------+----------+---------+------+------+----------+-----------------------+
| id | select_type | table | partitions | type | possible_keys | key | key_len | ref | rows | filtered | Extra || 1 | SIMPLE | tuser | NULL | range | na_index | na_index | 102 | NULL | 2 | 25.00 | Using index condition |

相关系统参数

索引条件下推默认是开启的,可以使用系统参数optimizer_switch来控制器是否开启。
查看默认状态:

 

切换状态:

set ="index_condition_pushdown=off";
set ="index_condition_pushdown=on";

思考

索引下推优化技术其实就是充分利用了索引中的数据,尽量在查询出整行数据之前过滤掉无效的数据。

由于需要存储引擎将索引中的数据与条件进行判断,所以这个技术是基于存储引擎的,只有特定引擎可以使用。并且判断条件需要是在存储引擎这个层面可以进行的操作才可以,比如调用存储过程的条件就不可以,因为存储引擎没有调用存储过程的能力。

参考:

1、《 MySQL技术内幕  InnoDB存储引擎》

2、《MySQL实战45讲》

  • 4
    点赞
  • 8
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: 索引下推是一种优化技术,它可以在查询过程中尽可能地利用索引来减少扫描的数据量,从而提高查询效率。具体来说,当一个查询包含多个条件时,索引下推可以将这些条件尽可能地推到索引层级中,以减少需要扫描的数据量。这样可以避免在查询过程中扫描大量的数据块,从而提高查询效率。 ### 回答2: 索引下推MySQL数据库中的一种优化技术,它在处理SQL查询时,首先使用索引来过滤掉不符合条件的行,然后再对符合条件的行进行进一步的判断和筛选。 传统的查询方式是,MySQL会先使用索引找到符合条件的行的主键,然后再通过主键去获取完整的行数据。这种方式需要查询两次磁盘或内存,会增加IO负载和查询时间。 而索引下推则是在执行索引查找的同时,将符合条件的数据进行过滤,减少了对磁盘或内存的读取和查询次数。它可以直接对索引进行访问,然后在索引访问路径上进行条件判断,只将符合条件的行返回给查询结果。这样可以减少对数据块的读取和查询次数,提高查询效率。 索引下推技术适用于多列查询条件的情况,例如同时使用了多个WHERE子句或使用了多个列的联合索引。它可以在索引的搜索路径上对多个查询条件进行判断,将不符合条件的行直接过滤掉,只返回满足所有条件的行。 需要注意的是,索引下推并非适用于所有情况,有些查询条件较为复杂或索引选择性较低时,索引下推的效果可能并不理想。此外,索引下推只适用于查找操作,对于更新操作并不适用。 综上所述,索引下推MySQL中的一种查询优化技术,通过在索引访问路径上对查询条件进行判断,减少数据块的读取和查询次数,从而提高查询效率。 ### 回答3: 索引下推MySQL优化查询性能的一种技术,它能够通过索引提前过滤掉不符合查询条件的行,从而减少了对数据的访问量,提高了查询效率。 在传统的查询过程中,MySQL首先会根据查询条件扫描索引,找到匹配的行,然后再针对这些行到数据表中进行查找。这种方式的问题在于,对于某些情况下,索引范围的扫描可能会导致大量不符合查询条件的行被读取出来,浪费了IO资源和CPU资源。 索引下推的目的就是在扫描索引的过程中,就根据查询条件对索引进行过滤,将不符合条件的行排除掉。这样一来,在访问数据表之前,就减少了对不符合条件的行的读取,从而提高了查询效率。 索引下推的原理是在索引节点上新增一个标记位,用于标记该节点下方的所有行是否符合查询条件。当查询中存在多个索引时,MySQL会根据查询条件选择合适的索引进行索引下推,并根据索引扫描的速度和是否过滤掉大量行数的能力进行判断。 需要注意的是,索引下推并不是适用于所有情况,只有在具备一定条件下才能发挥作用。具体来说,索引下推主要适用于范围条件查询、多列条件查询等场景。同时,索引下推只在MySQL优化器认为其可行且能提高查询性能的情况下才会被使用。 总结来说,索引下推MySQL中一种优化查询性能的技术,通过在扫描索引的过程中对索引进行过滤,减少不符合查询条件的行的读取,从而提高查询效率。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值