1229【MySQL】性能优化之 Index Condition Pushdown
转自http://blog.itpub.net/22664653/viewspace-1210844/
分类: MySQL
Index Condition Pushdown (ICP)是MySQL 5.6 版本中的新特性,是一种在存储引擎层使用索引过滤数据的一种优化方式。
a 当关闭ICP时,index 仅仅是data access 的一种访问方式,存储引擎通过索引回表获取的数据会传递到MySQL Server 层进行where条件过滤。
b 当打开ICP时,如果部分where条件能使用索引中的字段,MySQL Server 会把这部分下推到引擎层,可以利用index过滤的where条件在存储引擎层进行数据过滤,而非将所有通过index access的结果传递到MySQL server层进行where过滤.
优化效果:ICP能减少引擎层访问基表的次数和MySQL Server 访问存储引擎的次数,减少io次数,提高查询语句性能。
二 原理
Index Condition Pushdown is not used:
1 Get the next row, first by reading the index tuple, and then by using the index tuple to locate and read the full table row.
2 Test the part of the WHERE condition that applies to this table. Accept or reject the row based on the test result.
Index Condition Pushdown is used
1 Get the next row s index tuple (but not the full table row).
2 Test the part of the WHERE condition that applies to this table and can be checked using only index columns.
If the condition is not satisfied, proceed to the index tuple for the next row.
3 If the condition is satisfied, use the index tuple to locate and read the full table row.
4 est the remaining part of the WHERE condition that applies to this table. Accept or reject the row based on the test result.
三 实践案例
a 环境准备
数据库版本 5.6.16
关闭缓存
set query_cache_size=0;
set query_cache_type=OFF;
测试数据下载地址
b 当开启ICP时
- mysql> SET profiling = 1;
- Query OK, 0 rows affected, 1 warning (0.00 sec)
- mysql> select * from employees where first_name='Anneke' and last_name like '%sig' ;
- +--------+------------+------------+-----------+--------+------------+
- | emp_no | birth_date | first_name | last_name | gender | hire_date |
- +--------+------------+------------+-----------+--------+------------+
- | 10006 | 1953-04-20 | Anneke | Preusig | F | 1989-06-02 |
- +--------+------------+------------+-----------+--------+------------+
- 1 row in set (0.00 sec)
- mysql> show profiles;
- +----------+------------+--------------------------------------------------------------------------------+
- | Query_ID | Duration | Query |
- +----------+------------+--------------------------------------------------------------------------------+
- | 1 | 0.00060275 | select * from employees where first_name='Anneke' and last_name like '%sig' |
- +----------+------------+--------------------------------------------------------------------------------+
- 3 rows in set, 1 warning (0.00 sec)
此时情况下根据MySQL的最左前缀原则, first_name 可以使用索引,last_name采用了like 模糊查询,不能使用索引。
c 关闭ICP
- mysql> set optimizer_switch='index_condition_pushdown=off';
- Query OK, 0 rows affected (0.00 sec)
- mysql> SET profiling = 1;
- Query OK, 0 rows affected, 1 warning (0.00 sec)
- mysql> select * from employees where first_name='Anneke' and last_name like '%sig' ;
- +--------+------------+------------+-----------+--------+------------+
- | emp_no | birth_date | first_name | last_name | gender | hire_date |
- +--------+------------+------------+-----------+--------+------------+
- | 10006 | 1953-04-20 | Anneke | Preusig | F | 1989-06-02 |
- +--------+------------+------------+-----------+--------+------------+
- 1 row in set (0.00 sec)
- mysql> SET profiling = 0;
- Query OK, 0 rows affected, 1 warning (0.00 sec)
- mysql> show profiles;
- +----------+------------+--------------------------------------------------------------------------------+
- | Query_ID | Duration | Query |
- +----------+------------+--------------------------------------------------------------------------------+
- | 2 | 0.00097000 | select * from employees where first_name='Anneke' and last_name like '%sig' |
- +----------+------------+--------------------------------------------------------------------------------+
- 6 rows in set, 1 warning (0.00 sec)
当开启ICP时 查询在sending data环节时间消耗是 0.000189s
- mysql> show profile cpu,block io for query 1;
- +----------------------+----------+----------+------------+--------------+---------------+
- | Status | Duration | CPU_user | CPU_system | Block_ops_in | Block_ops_out |
- +----------------------+----------+----------+------------+--------------+---------------+
- | starting | 0.000094 | 0.000000 | 0.000000 | 0 | 0 |
- | checking permissions | 0.000011 | 0.000000 | 0.000000 | 0 | 0 |
- | Opening tables | 0.000025 | 0.000000 | 0.000000 | 0 | 0 |
- | init | 0.000044 | 0.000000 | 0.000000 | 0 | 0 |
- | System lock | 0.000014 | 0.000000 | 0.000000 | 0 | 0 |
- | optimizing | 0.000021 | 0.000000 | 0.000000 | 0 | 0 |
- | statistics | 0.000093 | 0.000000 | 0.000000 | 0 | 0 |
- | preparing | 0.000024 | 0.000000 | 0.000000 | 0 | 0 |
- | executing | 0.000006 | 0.000000 | 0.000000 | 0 | 0 |
- | Sending data | 0.000189 | 0.000000 | 0.000000 | 0 | 0 |
- | end | 0.000019 | 0.000000 | 0.000000 | 0 | 0 |
- | query end | 0.000012 | 0.000000 | 0.000000 | 0 | 0 |
- | closing tables | 0.000013 | 0.000000 | 0.000000 | 0 | 0 |
- | freeing items | 0.000034 | 0.000000 | 0.000000 | 0 | 0 |
- | cleaning up | 0.000007 | 0.000000 | 0.000000 | 0 | 0 |
- +----------------------+----------+----------+------------+--------------+---------------+
- 15 rows in set, 1 warning (0.00 sec)
当关闭ICP时 查询在sending data环节时间消耗是 0.000735s
- mysql> show profile cpu,block io for query 2;
- +----------------------+----------+----------+------------+--------------+---------------+
- | Status | Duration | CPU_user | CPU_system | Block_ops_in | Block_ops_out |
- +----------------------+----------+----------+------------+--------------+---------------+
- | starting | 0.000045 | 0.000000 | 0.000000 | 0 | 0 |
- | checking permissions | 0.000007 | 0.000000 | 0.000000 | 0 | 0 |
- | Opening tables | 0.000015 | 0.000000 | 0.000000 | 0 | 0 |
- | init | 0.000024 | 0.000000 | 0.000000 | 0 | 0 |
- | System lock | 0.000009 | 0.000000 | 0.000000 | 0 | 0 |
- | optimizing | 0.000012 | 0.000000 | 0.000000 | 0 | 0 |
- | statistics | 0.000049 | 0.000000 | 0.000000 | 0 | 0 |
- | preparing | 0.000016 | 0.000000 | 0.000000 | 0 | 0 |
- | executing | 0.000005 | 0.000000 | 0.000000 | 0 | 0 |
- | Sending data | 0.000735 | 0.001000 | 0.000000 | 0 | 0 |
- | end | 0.000008 | 0.000000 | 0.000000 | 0 | 0 |
- | query end | 0.000008 | 0.000000 | 0.000000 | 0 | 0 |
- | closing tables | 0.000009 | 0.000000 | 0.000000 | 0 | 0 |
- | freeing items | 0.000023 | 0.000000 | 0.000000 | 0 | 0 |
- | cleaning up | 0.000007 | 0.000000 | 0.000000 | 0 | 0 |
- +----------------------+----------+----------+------------+--------------+---------------+
- 15 rows in set, 1 warning (0.00 sec)
从上面的profile 可以看出ICP 开启时整个sql 执行时间是未开启的2/3,sending data 环节的时间消耗前者仅是后者的1/4。
- mysql> explain select * from employees where first_name='Anneke' and last_name like '%nta' ;
- +----+-------------+-----------+------+---------------+--------------+---------+-------+------+-----------------------+
- | id | select_type | table | type | possible_keys | key | key_len | ref | rows | Extra |
- +----+-------------+-----------+------+---------------+--------------+---------+-------+------+-----------------------+
- | 1 | SIMPLE | employees | ref | idx_emp_fnln | idx_emp_fnln | 44 | const | 224 | Using index condition |
- +----+-------------+-----------+------+---------------+--------------+---------+-------+------+-----------------------+
- 1 row in set (0.00 sec)
ICP 关闭时的执行计划显示use where.
- mysql> explain select * from employees where first_name='Anneke' and last_name like '%nta' ;
- +----+-------------+-----------+------+---------------+--------------+---------+-------+------+-------------+
- | id | select_type | table | type | possible_keys | key | key_len | ref | rows | Extra |
- +----+-------------+-----------+------+---------------+--------------+---------+-------+------+-------------+
- | 1 | SIMPLE | employees | ref | idx_emp_fnln | idx_emp_fnln | 44 | const | 224 | Using where |
- +----+-------------+-----------+------+---------------+--------------+---------+-------+------+-------------+
- 1 row in set (0.00 sec)
案例分析
以上面的查询为例关闭ICP 时,存储引擎通前缀index first_name 访问表中225条first_name 为Anneke的数据,并在MySQL server层根据last_name like '%sig' 进行过滤
开启ICP 时,last_name 的like '%sig'条件可以通过索引字段last_name 进行过滤,在存储引擎内部通过与where条件的对比,直接过滤掉不符合条件的数据。该过程不回表,只访问符合条件的1条记录并返回给MySQL Server ,有效的减少了io访问和各层之间的交互。
ICP 关闭时 ,仅仅使用索引作为访问数据的方式。
ICP 开启时 ,MySQL将在存储引擎层 利用索引过滤数据,减少不必要的回表,注意 虚线的using where 表示如果where条件中含有没有被索引的字段,则还是要经过MySQL Server 层过滤。
四 ICP的使用限制
1 当sql需要全表访问时,ICP的优化策略可用于range, ref, eq_ref, ref_or_null 类型的访问数据方法 。
2 支持InnoDB和MyISAM表。
3 ICP只能用于二级索引,不能用于主索引。
4 并非全部where条件都可以用ICP筛选。
如果where条件的字段不在索引列中,还是要读取整表的记录到server端做where过滤。
5 ICP的加速效果取决于在存储引擎内通过ICP筛选掉的数据的比例。
6 5.6 版本的不支持分表的ICP 功能,5.7 版本的开始支持。
7 当sql 使用覆盖索引时,不支持ICP 优化方法。
- mysql> explain select * from employees where first_name='Anneke' and last_name='Porenta' ;
- +----+-------------+-----------+------+---------------+--------------+---------+-------------+------+-----------------------+
- | id | select_type | table | type | possible_keys | key | key_len | ref | rows | Extra |
- +----+-------------+-----------+------+---------------+--------------+---------+-------------+------+-----------------------+
- | 1 | SIMPLE | employees | ref | idx_emp_fnln | idx_emp_fnln | 94 | const,const | 1 | Using index condition |
- +----+-------------+-----------+------+---------------+--------------+---------+-------------+------+-----------------------+
- 1 row in set (0.00 sec)
- mysql> explain select first_name,last_name from employees where first_name='Anneke' and last_name='Porenta' ;
- +----+-------------+-----------+------+---------------+--------------+---------+-------------+------+--------------------------+
- | id | select_type | table | type | possible_keys | key | key_len | ref | rows | Extra |
- +----+-------------+-----------+------+---------------+--------------+---------+-------------+------+--------------------------+
- | 1 | SIMPLE | employees | ref | idx_emp_fnln | idx_emp_fnln | 94 | const,const | 1 | Using where; Using index |
- +----+-------------+-----------+------+---------------+--------------+---------+-------------+------+--------------------------+
1229【MySQL】性能优化之 Index Condition Pushdown的更多相关文章
- 【MySQL】性能优化之 Index Condition Pushdown
一 概念介绍 Index Condition Pushdown (ICP)是MySQL 5.6 版本中的新特性,是一种在存储引擎层使用索引过滤数据的一种优化方式.a 当关闭ICP时,index ...
- 浅析MySQL中的Index Condition Pushdown (ICP 索引条件下推)和Multi-Range Read(MRR 索引多范围查找)查询优化
本文出处:http://www.cnblogs.com/wy123/p/7374078.html(保留出处并非什么原创作品权利,本人拙作还远远达不到,仅仅是为了链接到原文,因为后续对可能存在的一些错误 ...
- MySQL 中Index Condition Pushdown (ICP 索引条件下推)和Multi-Range Read(MRR 索引多范围查找)查询优化
一.ICP优化原理 Index Condition Pushdown (ICP),也称为索引条件下推,体现在执行计划的上是会出现Using index condition(Extra列,当然Extra ...
- MySQL 优化之 ICP (index condition pushdown:索引条件下推)
ICP技术是在MySQL5.6中引入的一种索引优化技术.它能减少在使用 二级索引 过滤where条件时的回表次数 和 减少MySQL server层和引擎层的交互次数.在索引组织表中,使用二级索引进行 ...
- MySQL 5.6 中一个重要的优化——Index Condition Pushdown,究竟push down了什么
1 问题描述 一条SQL,在数据库中是如何执行的呢?相信很多人都会对这个问题比较感兴趣.当然,要完整描述一条SQL在数据库中的生命周期,这是一个非常巨大的问题,涵盖了SQL的词法解析.语 ...
- MySQL索引与Index Condition Pushdown
实际上,这个页面所讲述的是在MariaDB 5.3.3(MySQL是在5.6)开始引入的一种叫做Index Condition Pushdown(以下简称ICP)的查询优化方式.由于本身不是一个层面的 ...
- MySQL ICP(Index Condition Pushdown)特性
一.SQL的where条件提取规则 在ICP(Index Condition Pushdown,索引条件下推)特性之前,必须先搞明白根据何登成大神总结出一套放置于所有SQL语句而皆准的where查询条 ...
- MySQL 之 Index Condition Pushdown(ICP)
简介 Index Condition Pushdown (ICP)是MySQL 5.6 版本中的新特性,是一种在存储引擎层使用索引过滤数据的一种优化方式. 当关闭ICP时,index 仅仅是data ...
- MySQL Index Condition Pushdown
Index Condition Pushdown (ICP)是MySQL 5.6 版本中的新特性,是一种在存储引擎层使用索引过滤数据的一种优化方式.[Index Condition Pushdown] ...
随机推荐
- angularjs学习使用分享
angularjs是一个为动态web应用设计的结构框架,它是为了克服html在构建应用上的不足而设计的. 工作原理: 1 加载html,然后解析成DOM: 2 加载angular.js脚本: 3 An ...
- C# 本质论 第二章 数据类型
浮点数的精度由有效数字的个数决定.除非用分数表示时,分母恰好是2的整数次幂,否则用二进制浮点类型无法准确地表示该数(0.1,表示成分数是1/10,分母10不能用有限二进制表示),二进制浮点类型无法准确 ...
- Xcode7.1环境下上架iOS App到AppStore 流程② (Part 二)
前言部分 part二部分主要讲解 iOS App IDs 的创建.概要文件的配置.以及概要文件安装的过程. 一.iOS App IDs 的创建 1)进入如图1所示界面点击右上角箭头所指的加号 进入iO ...
- Scala Macros - 元编程 Metaprogramming with Def Macros
Scala Macros对scala函数库编程人员来说是一项不可或缺的编程工具,可以通过它来解决一些用普通编程或者类层次编程(type level programming)都无法解决的问题,这是因为S ...
- (学习笔记)laravel 中间件
(学习笔记)laravel 中间件 laravel的请求在进入逻辑处理之前会通过http中间件进行处理. 也就是说http请求的逻辑是这样的: 建立中间件 首先,通过Artisan命令建立一个中间件. ...
- Lind.DDD.SSO单点登陆组件的使用(原创)
回到目录 一般sso的说明 在Lind.DDD框架里,有对单点登陆的集成,原理就是各个网站去sso网站统一登陆授权,之后在sso网站将登陆的token进行存储,存储方式随你(cache,redis,m ...
- Linux归档压缩、分区管理与LVM管理
归档和压缩命令: 命令格式: gzip [-9] 文件名 bzip2 [-9] 文件名 gzip –d .gz格式的压缩文件 bzip2 –d .bz2格式的压缩文件 选项: -9:高压缩比,多用于压 ...
- Web Worker javascript多线程编程(二)
Web Worker javascript多线程编程(一)中提到有两种Web Worker:专用线程dedicated web worker,以及共享线程shared web worker.不过主要讲 ...
- Linux2.6内核进程调度系列--scheduler_tick()函数2.更新实时进程的时间片
RT /** * 递减当前进程的时间片计数器,并检查是否已经用完时间片. * 由于进程的调度类型不同,函数所执行的操作也有很大差别. */ /* 如果是实时进程,就进一步根据是FIFO还是RR类型的实 ...
- RunLoop 总结:RunLoop的应用场景(二)
上一篇讲了使用RunLoop保证子线程的长时间存活,而不是执行完任务后就立刻销毁的应用场景.这一篇就讲述一下RunLoop如何保证NSTimer在视图滑动时,依然能正常运转. 参考资料 好的书籍都是值 ...