转载请注明来源:mysql索引合并:一条sql可以使用多个索引

前言

mysql的索引合并并不是什么新特性。早在mysql5.0版本就已经实现。之所以还写这篇博文,是因为好多人还一直保留着一条sql语句只能使用一个索引的错误观念。本文会通过一些示例来说明如何使用索引合并。

什么是索引合并

下面我们看下mysql文档中对索引合并的说明:The Index Merge method is used to retrieve rows with several range scans and to merge their results into one.

The merge can produce unions, intersections, or unions-of-intersections of its underlying scans.

This access method merges index scans from a single table;

it does not merge scans across multiple tables.

1、索引合并是把几个索引的范围扫描合并成一个索引。

2、索引合并的时候,会对索引进行并集,交集或者先交集再并集操作,以便合并成一个索引。

3、这些需要合并的索引只能是一个表的。不能对多表进行索引合并。

使用索引合并有啥收益

简单的说,索引合并,让一条sql可以使用多个索引。对这些索引取交集,并集,或者先取交集再取并集。从而减少从数据表中取数据的次数,提高查询效率。

怎么确定使用了索引合并

在使用explain对sql语句进行操作时,如果使用了索引合并,那么在输出内容的type列会显示 index_merge,key列会显示出所有使用的索引。如下:

在explain的extra字段中会以下几种:

Using union 索引取并集

Using sort_union 先对取出的数据按rowid排序,然后再取并集

Using intersect 索引取交集

你会发现并没有 sort_intersect,因为根据目前的实现,想索引取交集,必须保证通过索引取出的数据顺序和rowid顺序是一致的。所以,也就没必要sort了。

sort_union索引合并的示例

数据表结构

1

2

3

4

5

6

7

8

9

10

11

12

13

14

mysql> show create table test\G

*************************** 1. row ***************************

Table: test

Create Table: CREATE TABLE `test` (

`id` int(11) NOT NULL AUTO_INCREMENT,

`key1_part1` int(11) NOT NULL DEFAULT '0',

`key1_part2` int(11) NOT NULL DEFAULT '0',

`key2_part1` int(11) NOT NULL DEFAULT '0',

`key2_part2` int(11) NOT NULL DEFAULT '0',

PRIMARY KEY (`id`),

KEY `key1` (`key1_part1`,`key1_part2`),

KEY `key2` (`key2_part1`,`key2_part2`)

) ENGINE=MyISAM AUTO_INCREMENT=18 DEFAULT CHARSET=utf8

1 row in set (0.00 sec)

数据

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26

27

28

29

30

31

32

33

34

35

mysql> select * from test;

+----+------------+------------+------------+------------+

| id | key1_part1 | key1_part2 | key2_part1 | key2_part2 |

+----+------------+------------+------------+------------+

| 1 | 1 | 1 | 1 | 1 |

| 2 | 1 | 1 | 2 | 1 |

| 3 | 1 | 1 | 2 | 2 |

| 4 | 1 | 1 | 3 | 2 |

| 5 | 1 | 1 | 3 | 3 |

| 6 | 1 | 1 | 4 | 3 |

| 7 | 1 | 1 | 4 | 4 |

| 8 | 1 | 1 | 5 | 4 |

| 9 | 1 | 1 | 5 | 5 |

| 10 | 2 | 1 | 1 | 1 |

| 11 | 2 | 2 | 1 | 1 |

| 12 | 3 | 2 | 1 | 1 |

| 13 | 3 | 3 | 1 | 1 |

| 14 | 4 | 3 | 1 | 1 |

| 15 | 4 | 4 | 1 | 1 |

| 16 | 5 | 4 | 1 | 1 |

| 17 | 5 | 5 | 1 | 1 |

| 18 | 5 | 5 | 3 | 3 |

| 19 | 5 | 5 | 3 | 1 |

| 20 | 5 | 5 | 3 | 2 |

| 21 | 5 | 5 | 3 | 4 |

| 22 | 6 | 6 | 3 | 3 |

| 23 | 6 | 6 | 3 | 4 |

| 24 | 6 | 6 | 3 | 5 |

| 25 | 6 | 6 | 3 | 6 |

| 26 | 6 | 6 | 3 | 7 |

| 27 | 1 | 1 | 3 | 6 |

| 28 | 1 | 2 | 3 | 6 |

| 29 | 1 | 3 | 3 | 6 |

+----+------------+------------+------------+------------+

29 rows in set (0.00 sec)

使用索引合并的案例

1

2

3

4

5

6

7

8

9

10

11

12

13

mysql> explain select * from test where (key1_part1=4 and key1_part2=4) or (key2_part1=4 and key2_part2=4)\G

*************************** 1. row ***************************

id: 1

select_type: SIMPLE

table: test

type: index_merge

possible_keys: key1,key2

key: key1,key2

key_len: 8,4

ref: NULL

rows: 3

Extra: Using sort_union(key1,key2); Using where

1 row in set (0.00 sec)

未使用索引合并的案例

1

2

3

4

5

6

7

8

9

10

11

12

13

mysql> explain select * from test where (key1_part1=1 and key1_part2=1) or key2_part1=4\G

*************************** 1. row ***************************

id: 1

select_type: SIMPLE

table: test

type: ALL

possible_keys: key1,key2

key: NULL

key_len: NULL

ref: NULL

rows: 29

Extra: Using where

1 row in set (0.00 sec)

sort_union总结

从上面的两个案例大家可以发现,相同模式的sql语句,可能有时能使用索引,有时不能使用索引。是否能使用索引,取决于mysql查询优化器对统计数据分析后,是否认为使用索引更快。

因此,单纯的讨论一条sql是否可以使用索引有点片面,还需要考虑数据。

union索引合并使用案例

数据表结构

1

2

3

4

5

6

7

8

9

10

11

12

13

14

mysql> show create table test\G

*************************** 1. row ***************************

Table: test

Create Table: CREATE TABLE `test` (

`id` int(11) NOT NULL AUTO_INCREMENT,

`key1_part1` int(11) NOT NULL DEFAULT '0',

`key1_part2` int(11) NOT NULL DEFAULT '0',

`key2_part1` int(11) NOT NULL DEFAULT '0',

`key2_part2` int(11) NOT NULL DEFAULT '0',

PRIMARY KEY (`id`),

KEY `key1` (`key1_part1`,`key1_part2`,`id`),

KEY `key2` (`key2_part1`,`key2_part2`,`id`)

) ENGINE=InnoDB AUTO_INCREMENT=30 DEFAULT CHARSET=utf8

1 row in set (0.00 sec)

数据结构和之前有所调整。主要调整有如下两方面:

1、引擎从myisam改为了innodb。

2、组合索引中增加了id,并把id放在最后。

数据

数据和上面的数据一样。

使用索引合并的案例

1

2

3

4

5

6

7

8

9

10

11

12

13

mysql> explain select * from test where (key1_part1=4 and key1_part2=4) or (key2_part1=4 and key2_part2=4)\G

*************************** 1. row ***************************

id: 1

select_type: SIMPLE

table: test

type: index_merge

possible_keys: key1,key2

key: key1,key2

key_len: 8,8

ref: NULL

rows: 2

Extra: Using union(key1,key2); Using where

1 row in set (0.00 sec)

union总结

相同的数据,相同的sql语句,只是数据表结构有所调整,就从sort_union变为了union。有以下几个原因:

1、只要通过索引取出的数据已经按rowid进行了排序,就可以使用union。

2、组合索引中在最后加id字段,目的就是通过索引前两个字段取出的数据是按id排序。

3、把引擎从myisam改为innodb,目的就是让id和rowid的顺序一致。

intersect使用案例

mysql索引合并:一条sql可以使用多个索引

http://www.php.cn/

以上就是mysql索引合并:一条sql可以使用多个索引的内容,更多相关内容请关注PHP中文网(www.php.cn)!

本文原创发布php中文网,转载请注明出处,感谢您的尊重!

mysql使用多个索引_mysql索引合并:一条sql可以使用多个索引相关推荐

  1. mysql 可以用多个索引_mysql索引合并:一条sql可以使用多个索引

    前言 mysql的索引合并并不是什么新特性.早在mysql5.0版本就已经实现.之所以还写这篇博文,是因为好多人还一直保留着一条sql语句只能使用一个索引的错误观念.本文会通过一些示例来说明如何使用索 ...

  2. MySQL 一条sql添加多个索引或添加多个列(字段)

    MySQL 一条sql添加多个索引或添加多个列(字段) MySQL 一次添加多个索引 ALTER TABLE  table_name ADD INDEX idx1 ( `aaa`), ADD INDE ...

  3. mysql中的强制索引_MYSQL中常用的强制性操作(例如强制索引)

    mysql常用的hint 对于经常使用oracle的朋友可能知道,oracle的hint功能种类很多,对于优化sql语句提供了很多方法.同样,在mysql里,也有类似的hint功能.下面介绍一些常用的 ...

  4. MySQL中会用到age字段的索引_MySQL学习笔记(四):正确使用索引(二)

    上一篇学习到了MySQL中的索引是什么,以及一些关于索引的简单的知识,这篇我打算详细地谈一谈如何正确地使用索引.索引的使用看起来很简单,但是在真正去学习它的时候才发现,原来是暗藏玄机.使用索引时有很多 ...

  5. mysql语句在哪编写_mysql常见问题七:编写sql语句

    一.删除除了学号字段以外,其它字段都相同的冗余记录,只保留一条!(也就是要删除凤姐和田七中一条重复数据只留一条) 要求结果数据: 原始数据: CREATE TABLEtbl_students ( id ...

  6. mysql统计数量函数方法_mySql关于统计数量的SQL查询操作

    mySql关于统计数量的SQL查询操作,状态,订单,语句,函数,数量 mySql关于统计数量的SQL查询操作 易采站长站,站长之家为您整理了mySql关于统计数量的SQL查询操作的相关内容. 我就废话 ...

  7. mysql怎么查看刷脏页慢_一条SQL查询语句极为缓慢,如何去优化呢

    一条 SQL 查询语句执行的很慢,那是每次查询都很慢呢?还是大多数情况下是正常的,偶尔出现很慢呢?可以分以下两种情况来讨论. 大多数情况是正常的,只是偶尔会出现很慢的情况. 在数据量不变的情况下,这条 ...

  8. mysql 5.6 5.7不兼容_同一条sql在mysql5.6和5.7版本遇到的问题。

    之前用的是mysql 5.6版本,执行select * from table group by colunm 是可以出结果的, 但是切换的5.7版本,这条sql就报错, Expression #1 o ...

  9. mysql 5.6 5.7不兼容_同一条sql在MySQL数据库5.6和5.7版本遇到的问题

    本文主要向大家介绍了同一条sql在MySQL数据库5.6和5.7版本遇到的问题 ,通过具体的内容向大家展现,希望对大家学习MySQL数据库有所帮助. 之前用的是mysql 5.6版本,执行select ...

  10. mysql 同样字段的索引_MySQL表中具有相同字段的两个索引

    MySQL 5.7.9 – 删除id_type索引并没有什么区别.多列索引(type_date)适用于两个查询. 解释查询输出: mysql> explain SELECT id_type,da ...

最新文章

  1. 机器学习(6)KNN算法(K-近邻算法)
  2. 随办 企业打造完美执行团队的终极利器
  3. ERROR 1044 (42000)报错的解决
  4. OCP-Oracle 12c 数据库认证
  5. [故障公告]14:40-15:00博客站点web服务器雪崩似的CPU 100%
  6. 【一起去大厂系列】针对left join以及limit的两条优化小技巧
  7. 动态T-SQL语句常見問題與解決方案
  8. [收藏转载]C# GDI+ 简单绘图(一)
  9. 小程序直播 OBS 画质_微信小程序怎么直播卖货?
  10. 修改oracle重做日志文件大小
  11. 浅析ReentrantReadWriteLock读写锁
  12. as3程序主类,执行顺序
  13. java 重量单位换算_用java创建一个可以换算长度单位的length类,将某种长度单位进行数值间的转换。...
  14. 《丁丁历险记系列之委托》改编自《.NET委托:一个C#睡前故事》
  15. javaweb课程设计在线学习论坛
  16. 塑料成型计算机模拟技术,塑料成型模拟软件技术基础与应用
  17. 基于matlab的心电信号预处理
  18. OpenNLP入门实验
  19. 已知销售额怎么计算成本_计算成本根据销售额怎么样推算出成本,举例, – 手机爱问...
  20. 先有鸡还有现有的蛋的终极答案

热门文章

  1. 2018年度中国LC3盛大召开!
  2. ajax中sy,黑马eesy_15 Vue:vue语法和生命周期与ajax异步请求
  3. 【优化算法】白冠鸡优化算法(COOT)【含Matlab源码 1795期】
  4. 【配送路径规划】基于matlab遗传算法求解带时间窗的多配送中心半开放式车辆路径规划问题【含Matlab源码 YC002期】
  5. ai人工智能_AI如何帮助截肢者?
  6. mnist 转图像_解决MNIST图像分类问题
  7. # 畸变矫正_拒绝拍照边缘畸变 坚果Pro 3 Smartisan OS v7.2.0系统体验
  8. android横向滑动分享,Android将ListView逆时针旋转90°实现横向滑动
  9. fatal error: libmemcached/memcached.h: No such file or directory
  10. 360浏览器:一键恢复不小心关闭的网页