热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

mysql5.6icpmrrbak_ICP、MRR、BKA等特性

ICP的目标是减少从基表中读取操作的数量,从而降低IO操作对于InnoDB表,ICP只适用于辅助索引当使用ICP优化时,执行计划的Extr

ICP的目标是减少从基表中读取操作的数量,从而降低IO操作

对于InnoDB表,ICP只适用于辅助索引

当使用ICP优化时,执行计划的Extra列显示Using indexcondition提示

数据库配置 optimizer_switch="index_condition_pushdown=on”;

使用场景举例

辅助索引INDEX (a, b, c)

SELECT * FROM peopleWHERE a='12345' AND b LIKE '%xx%'AND c LIKE '%yy%';

若不使用ICP:则是通过二级索引中a的值去基表取出所有a='12345'的数据,然后server层再对b LIKE '%xx%'AND c LIKE '%yy%' 进行过滤

若使用ICP:则b LIKE '%xx%'AND c LIKE '%yy%'的过滤操作在二级索引中完成,然后再去基表取相关数据

ICP特点

mysql 5.6中只支持 MyISAM、InnoDB、NDB cluster

mysql 5.6中不支持分区表的ICP,从MySQL 5.7.3开始支持分区表的ICP

ICP的优化策略可用于range、ref、eq_ref、ref_or_null 类型的访问数据方法

不支持主建索引的ICP(对于Innodb的聚集索引,完整的记录已经被读取到Innodb Buffer,此时使用ICP并不能降低IO操作)

当 SQL 使用覆盖索引时但只检索部分数据时,ICP 无法使用

ICP的加速效果取决于在存储引擎内通过ICP筛选掉的数据的比例

二、Multi-Range Read (MRR)

MRR 的全称是 Multi-Range Read Optimization,是优化器将随机 IO 转化为顺序 IO 以降低查询过程中 IO 开销的一种手段,这对IO-bound类型的SQL语句性能带来极大的提升,适用于range ref eq_ref类型的查询

MRR优化的几个好处

使数据访问有随机变为顺序,查询辅助索引是,首先把查询结果按照主键进行排序,按照主键的顺序进行书签查找

减少缓冲池中页被替换的次数

批量处理对键值的操作

在没有使用MRR特性时

第一步 先根据where条件中的辅助索引获取辅助索引与主键的集合,结果集为rest

select key_column, pk_column from tb where key_column=x order by key_column

第二步 通过第一步获取的主键来获取对应的值

for each pk_column value in rest do:

select non_key_column from tb where pk_column=val

使用MRR特性时

第一步 先根据where条件中的辅助索引获取辅助索引与主键的集合,结果集为rest

select key_column, pk_column from tb where key_column = x order by key_column

第二步 将结果集rest放在buffer里面(read_rnd_buffer_size 大小直到buffer满了),然后对结果集rest按照pk_column排序,得到结果集是rest_sort

第三步 利用已经排序过的结果集,访问表中的数据,此时是顺序IO.

select non_key_column fromtb where pk_column in (rest_sort)

在不使用 MRR 时,优化器需要根据二级索引返回的记录来进行“回表”,这个过程一般会有较多的随机IO, 使用MRR时,SQL语句的执行过程是这样的:

优化器将二级索引查询到的记录放到一块缓冲区中

如果二级索引扫描到文件的末尾或者缓冲区已满,则使用快速排序对缓冲区中的内容按照主键进行排序

用户线程调用MRR接口取cluster index,然后根据cluster index 取行数据

当根据缓冲区中的 cluster index取完数据,则继续调用过程 2) 3),直至扫描结束

通过上述过程,优化器将二级索引随机的 IO 进行排序,转化为主键的有序排列,从而实现了随机 IO 到顺序 IO 的转化,提升性能

此外MRR还可以将某些范围查询,拆分为键值对,来进行批量的数据查询,如下:

SELECT * FROM t WHERE key_part1>&#61; 1000 AND key_part1<2000AND key_part2&#61; 10000;

表t上有二级索引(key_part1, key_part2)&#xff0c;索引根据key_part1,key_part2的顺序排序。

若不使用MRR&#xff1a;此时查询的类型为Range&#xff0c;sql优化器会先将key_part1大于1000小于2000的数据取出&#xff0c;即使key_part2不等于10000&#xff0c;带取出之后再进行过滤&#xff0c;会导致很多无用的数据被取出

若使用MRR&#xff1a;如果索引中key_part2不为10000的元组越多&#xff0c;最终MRR的效果越好。优化器会将查询条件拆分为(1000,1000)&#xff0c;(1001,1000)&#xff0c;... (1999,1000)最终会根据这些条件进行过滤

相关参数

当mrr&#61;on,mrr_cost_based&#61;on&#xff0c;则表示cost base的方式还选择启用MRR优化,当发现优化后的代价过高时就会不使用该项优化

当mrr&#61;on,mrr_cost_based&#61;off&#xff0c;则表示总是开启MRR优化

SET &#64;&#64;optimizer_switch&#61;&#39;mrr&#61;on,mrr_cost_based&#61;on&#39;;

参数read_rnd_buffer_size 用来控制键值缓冲区的大小。二级索引扫描到文件的末尾或者缓冲区已满&#xff0c;则使用快速排序对缓冲区中的内容按照主键进行排序

三、Batched Key Access (BKA) 和 Block Nested-Loop(BNL)

Batched Key Access (BKA)  提高表 join 性能的算法。当被join的表能够使用索引时&#xff0c;就先排好顺序&#xff0c;然后再去检索被join的表&#xff0c;听起来和MRR类似&#xff0c;实际上MRR也可以想象成二级索引和 primary key的join

如果被Join的表上没有索引&#xff0c;则使用老版本的BNL策略(BLOCK Nested-loop)

BKA原理

对于多表join语句&#xff0c;当MySQL使用索引访问第二个join表的时候&#xff0c;使用一个join buffer来收集第一个操作对象生成的相关列值。BKA构建好key后&#xff0c;批量传给引擎层做索引查找。key是通过MRR接口提交给引擎的(mrr目的是较为顺序)MRR使得查询更有效率。

大致的过程如下:

BKA使用join buffer保存由join的第一个操作产生的符合条件的数据

然后BKA算法构建key来访问被连接的表&#xff0c;并批量使用MRR接口提交keys到数据库存储引擎去查找查找。

提交keys之后&#xff0c;MRR使用最佳的方式来获取行并反馈给BKA

BNL和BKA都是批量的提交一部分行给被join的表&#xff0c;从而减少访问的次数&#xff0c;那么它们有什么区别呢&#xff1f;

BNL比BKA出现的早&#xff0c;BKA直到5.6才出现&#xff0c;而NBL至少在5.1里面就存在。

BNL主要用于当被join的表上无索引

BKA主要是指在被join表上有索引可以利用&#xff0c;那么就在行提交给被join的表之前&#xff0c;对这些行按照索引字段进行排序&#xff0c;因此减少了随机IO&#xff0c;排序这才是两者最大的区别&#xff0c;但是如果被join的表没用索引呢&#xff1f;那就使用NBL

BKA和BNL标识

Using join buffer (Batched Key Access)和Using join buffer (Block Nested Loop)

相关参数

BAK使用了MRR&#xff0c;要想使用BAK必须打开MRR功能&#xff0c;而MRR基于mrr_cost_based的成本估算并不能保证总是使用MRR&#xff0c;官方推荐设置mrr_cost_based&#61;off来总是开启MRR功能。打开BAK功能(BAK默认OFF)&#xff1a;

SET optimizer_switch&#61;&#39;mrr&#61;on,mrr_cost_based&#61;off,batched_key_access&#61;on&#39;;

BKA使用join buffer size来确定buffer的大小&#xff0c;buffer越大&#xff0c;访问被join的表/内部表就越顺序。

BNL默认是开启的&#xff0c;设置BNL相关参数&#xff1a;

SET optimizer_switch&#61;’block_nested_loop’

支持inner join, outer join, semi-join operations,including nested outer joins

BKA主要适用于join的表上有索引可利用&#xff0c;无索引只能使用BNL

四、总结

ICP(Index Condition Pushdown)

Index Condition Pushdown是用索引去表里取数据的一种优化&#xff0c;减少了引擎层访问基表的次数和Server层访问存储引擎的次数&#xff0c;在引擎层就能够过滤掉大量的数据&#xff0c;减少io次数&#xff0c;提高查询语句性能

MRR(Multi-Range Read)

是基于辅助/第二索引的查询&#xff0c;减少随机IO&#xff0c;并且将随机IO转化为顺序IO&#xff0c;提高查询效率。

不使用MRR之前(MySQL5.6之前)&#xff0c;先根据where条件中的辅助索引获取辅助索引与主键的集合&#xff0c;再通过主键来获取对应的值。辅助索引获取的主键来访问表中的数据会导致随机的IO(辅助索引的存储顺序并非与主键的顺序一致)&#xff0c;随机主键不在同一个page里时会导致多次IO和随机读。

使用MRR优化(MySQL5.6之后)&#xff0c;先根据where条件中的辅助索引获取辅助索引与主键的集合&#xff0c;再将结果集放在buffer(read_rnd_buffer_size 直到buffer满了)&#xff0c;然后对结果集按照pk_column排序&#xff0c;得到有序的结果集rest_sort。最后利用已经排序过的结果集&#xff0c;访问表中的数据&#xff0c;此时是顺序IO。即MySQL 将根据辅助索引获取的结果集根据主键进行排序&#xff0c;将无序化为有序&#xff0c;可以用主键顺序访问基表&#xff0c;将随机读转化为顺序读&#xff0c;多页数据记录可一次性读入或根据此次的主键范围分次读入&#xff0c;减少IO操作&#xff0c;提高查询效率。



推荐阅读
  • MyBatis多表查询与动态SQL使用
    本文介绍了MyBatis多表查询与动态SQL的使用方法,包括一对一查询和一对多查询。同时还介绍了动态SQL的使用,包括if标签、trim标签、where标签、set标签和foreach标签的用法。文章还提供了相关的配置信息和示例代码。 ... [详细]
  • 图解redis的持久化存储机制RDB和AOF的原理和优缺点
    本文通过图解的方式介绍了redis的持久化存储机制RDB和AOF的原理和优缺点。RDB是将redis内存中的数据保存为快照文件,恢复速度较快但不支持拉链式快照。AOF是将操作日志保存到磁盘,实时存储数据但恢复速度较慢。文章详细分析了两种机制的优缺点,帮助读者更好地理解redis的持久化存储策略。 ... [详细]
  • vue使用
    关键词: ... [详细]
  • 一、Hadoop来历Hadoop的思想来源于Google在做搜索引擎的时候出现一个很大的问题就是这么多网页我如何才能以最快的速度来搜索到,由于这个问题Google发明 ... [详细]
  • Linux服务器密码过期策略、登录次数限制、私钥登录等配置方法
    本文介绍了在Linux服务器上进行密码过期策略、登录次数限制、私钥登录等配置的方法。通过修改配置文件中的参数,可以设置密码的有效期、最小间隔时间、最小长度,并在密码过期前进行提示。同时还介绍了如何进行公钥登录和修改默认账户用户名的操作。详细步骤和注意事项可参考本文内容。 ... [详细]
  • 本文介绍了如何在MySQL中将零值替换为先前的非零值的方法,包括使用内联查询和更新查询。同时还提供了选择正确值的方法。 ... [详细]
  • IhaveconfiguredanactionforaremotenotificationwhenitarrivestomyiOsapp.Iwanttwodiff ... [详细]
  • 本文介绍了如何使用php限制数据库插入的条数并显示每次插入数据库之间的数据数目,以及避免重复提交的方法。同时还介绍了如何限制某一个数据库用户的并发连接数,以及设置数据库的连接数和连接超时时间的方法。最后提供了一些关于浏览器在线用户数和数据库连接数量比例的参考值。 ... [详细]
  • 使用Ubuntu中的Python获取浏览器历史记录原文: ... [详细]
  • sklearn数据集库中的常用数据集类型介绍
    本文介绍了sklearn数据集库中常用的数据集类型,包括玩具数据集和样本生成器。其中详细介绍了波士顿房价数据集,包含了波士顿506处房屋的13种不同特征以及房屋价格,适用于回归任务。 ... [详细]
  • 高质量SQL书写的30条建议
    本文提供了30条关于优化SQL的建议,包括避免使用select *,使用具体字段,以及使用limit 1等。这些建议是基于实际开发经验总结出来的,旨在帮助读者优化SQL查询。 ... [详细]
  • 本文讨论了在数据库打开和关闭状态下,重新命名或移动数据文件和日志文件的情况。针对性能和维护原因,需要将数据库文件移动到不同的磁盘上或重新分配到新的磁盘上的情况,以及在操作系统级别移动或重命名数据文件但未在数据库层进行重命名导致报错的情况。通过三个方面进行讨论。 ... [详细]
  • Oracle seg,V$TEMPSEG_USAGE与Oracle排序的关系及使用方法
    本文介绍了Oracle seg,V$TEMPSEG_USAGE与Oracle排序之间的关系,V$TEMPSEG_USAGE是V_$SORT_USAGE的同义词,通过查询dba_objects和dba_synonyms视图可以了解到它们的详细信息。同时,还探讨了V$TEMPSEG_USAGE的使用方法。 ... [详细]
  • 本文介绍了Android 7的学习笔记总结,包括最新的移动架构视频、大厂安卓面试真题和项目实战源码讲义。同时还分享了开源的完整内容,并提醒读者在使用FileProvider适配时要注意不同模块的AndroidManfiest.xml中配置的xml文件名必须不同,否则会出现问题。 ... [详细]
  • 前景:当UI一个查询条件为多项选择,或录入多个条件的时候,比如查询所有名称里面包含以下动态条件,需要模糊查询里面每一项时比如是这样一个数组条件:newstring[]{兴业银行, ... [详细]
author-avatar
W14154988
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有