阿里云-云小站(无限量代金券发放中)
【腾讯云】云服务器、云数据库、COS、CDN、短信等热卖云产品特惠抢购

MySQL 5.7使用or是否会用到索引并提高查询效率的理解

428次阅读
没有评论

共计 8116 个字符,预计需要花费 21 分钟才能阅读完成。

相信很多人在 MySQL 中看到了 where 条件中使用到了 or 就会以为这样是不会走索引的,通常会使用 union all 或者 in 来进行优化,事实并不是想象的这样具体问题具体分析。

下面我们来看看

首先我们用 sysbench 生成两个 100w 行的表

表结构如下

mysql> show create table sbtest1 \G;
*************************** 1. row ***************************
      Table: sbtest1
Create Table: CREATE TABLE `sbtest1` (
  `id` int(11) NOT NULL AUTO_INCREMENT,
  `k` int(11) NOT NULL DEFAULT ‘0’,
  `c` char(120) NOT NULL DEFAULT ”,
  `pad` char(60) NOT NULL DEFAULT ”,
  PRIMARY KEY (`id`),
  KEY `k_1` (`k`),
  KEY `c_1` (`c`)
) ENGINE=InnoDB AUTO_INCREMENT=1000001 DEFAULT CHARSET=latin1
1 row in set (0.00 sec)

ERROR:
No query specified

mysql> show create table sbtest2 \G;
*************************** 1. row ***************************
      Table: sbtest2
Create Table: CREATE TABLE `sbtest2` (
  `id` int(11) NOT NULL AUTO_INCREMENT,
  `k` int(11) NOT NULL DEFAULT ‘0’,
  `c` char(120) NOT NULL DEFAULT ”,
  `pad` char(60) NOT NULL DEFAULT ”,
  PRIMARY KEY (`id`),
  KEY `k_2` (`k`),
  KEY `c_2` (`c`)
) ENGINE=InnoDB AUTO_INCREMENT=1000001 DEFAULT CHARSET=latin1
1 row in set (0.00 sec)

ERROR:
No query specified

1. 首先我们使用同一列带索引字段的进行查询。

mysql> explain select * from sbtest1 where k=’501462′ or k=’502480′; 
+—-+————-+———+————+——-+—————+——+———+——+——+———-+———————–+
| id | select_type | table  | partitions | type  | possible_keys | key  | key_len | ref  | rows | filtered | Extra                |
+—-+————-+———+————+——-+—————+——+———+——+——+———-+———————–+
|  1 | SIMPLE      | sbtest1 | NULL      | range | k_1          | k_1  | 4      | NULL |  214 |  100.00 | Using index condition |
+—-+————-+———+————+——-+—————+——+———+——+——+———-+———————–+

从执行计划中看出这样是可以使用到索引的,另外我们使用 in 或者 union all 来看。

mysql> explain select pad from sbtest1 where k in (‘501462′,’502480’);
+—-+————-+———+————+——-+—————+——+———+——+——+———-+———————–+
| id | select_type | table  | partitions | type  | possible_keys | key  | key_len | ref  | rows | filtered | Extra                |
+—-+————-+———+————+——-+—————+——+———+——+——+———-+———————–+
|  1 | SIMPLE      | sbtest1 | NULL      | range | k_1          | k_1  | 4      | NULL |  214 |  100.00 | Using index condition |
+—-+————-+———+————+——-+—————+——+———+——+——+———-+———————–+

in 的执行计划和 or 相同。

mysql>  explain select pad from sbtest1 where k=’501462′ union all select pad from sbtest1 where k=’502480′;
+—-+————-+———+————+——+—————+——+———+——-+——+———-+——-+
| id | select_type | table  | partitions | type | possible_keys | key  | key_len | ref  | rows | filtered | Extra |
+—-+————-+———+————+——+—————+——+———+——-+——+———-+——-+
|  1 | PRIMARY    | sbtest1 | NULL      | ref  | k_1          | k_1  | 4      | const |  113 |  100.00 | NULL  |
|  2 | UNION      | sbtest1 | NULL      | ref  | k_1          | k_1  | 4      | const |  101 |  100.00 | NULL  |

虽然执行计划不通但 union all 估计的查询行数和上面相同。

2. 我们再来看看不同列带索引字段的进行查询

mysql> explain select pad from sbtest1 where  k=’501462′ or c=’68487932199-96439406143-93774651418-41631865787-96406072701-20604855487-25459966574-28203206787-41238978918-19503783441′;
+—-+————-+———+————+————-+—————+———+———+——+——+———-+———————————–+
| id | select_type | table  | partitions | type        | possible_keys | key    | key_len | ref  | rows | filtered | Extra                            |
+—-+————-+———+————+————-+—————+———+———+——+——+———-+———————————–+
|  1 | SIMPLE      | sbtest1 | NULL      | index_merge | k_1,c_1      | k_1,c_1 | 4,120  | NULL |  114 |  100.00 | Using union(k_1,c_1); Using where |
+—-+————-+———+————+————-+—————+———+———+——+——+———-+———————————–

这样的情况也会使用索引

如果 or 的条件中有个条件不带索引的话,那这条 sql 就不会使用到索引了,如下。

mysql> explain select pad from sbtest1 where  k=’501462′ or pad=’00592560354-80393027097-78244247549-39135306455-88936868384′;
+—-+————-+———+————+——+—————+——+———+——+——–+———-+————-+
| id | select_type | table  | partitions | type | possible_keys | key  | key_len | ref  | rows  | filtered | Extra      |
+—-+————-+———+————+——+—————+——+———+——+——–+———-+————-+
|  1 | SIMPLE      | sbtest1 | NULL      | ALL  | k_1          | NULL | NULL    | NULL | 986400 |    19.00 | Using where |
+—-+————-+———+————+——+—————+——+———+——+——–+———-+————-+

pad 列没索引所以整条的 sql 就不会使用到索引

假设使用 union all 来改写一样需要全表扫描所以意义也不大,如下

mysql>  explain select pad from sbtest1 where  k=’501462′ union all select pad from sbtest1 where pad=’00592560354-80393027097-78244247549-39135306455-88936868384′;
+—-+————-+———+————+——+—————+——+———+——-+——–+———-+————-+
| id | select_type | table  | partitions | type | possible_keys | key  | key_len | ref  | rows  | filtered | Extra      |
+—-+————-+———+————+——+—————+——+———+——-+——–+———-+————-+
|  1 | PRIMARY    | sbtest1 | NULL      | ref  | k_1          | k_1  | 4      | const |    113 |  100.00 | NULL        |
|  2 | UNION      | sbtest1 | NULL      | ALL  | NULL          | NULL | NULL    | NULL  | 986400 |    10.00 | Using where |
+—-+————-+———+————+——+—————+——+———+——-+——–+———-+————-+

3. 接下来我们看看多表关联查询

mysql> explain select a.pad,b.pad from sbtest1 a,sbtest2 b where a.id=b.id and (a.c=’123′ or b.c=’1234′);               
+—-+————-+——-+————+——–+—————+———+———+———–+——–+———-+————-+
| id | select_type | table | partitions | type  | possible_keys | key    | key_len | ref      | rows  | filtered | Extra      |
+—-+————-+——-+————+——–+—————+———+———+———–+——–+———-+————-+
|  1 | SIMPLE      | a    | NULL      | ALL    | PRIMARY,c_1  | NULL    | NULL    | NULL      | 986400 |  100.00 | NULL        |
|  1 | SIMPLE      | b    | NULL      | eq_ref | PRIMARY,c_2  | PRIMARY | 4      | test.a.id |      1 |  100.00 | Using where |
+—-+————-+——-+————+——–+—————+———+———+———–+——–+———-+————-+
2 rows in set, 1 warning (0.00 sec)

mysql> explain select a.pad,b.pad from sbtest1 a,sbtest2 b where a.id=b.id and (a.c=’123′ or a.c=’1234′);
+—-+————-+——-+————+——–+—————+———+———+———–+——+———-+———————–+
| id | select_type | table | partitions | type  | possible_keys | key    | key_len | ref      | rows | filtered | Extra                |
+—-+————-+——-+————+——–+—————+———+———+———–+——+———-+———————–+
|  1 | SIMPLE      | a    | NULL      | range  | PRIMARY,c_1  | c_1    | 120    | NULL      |    2 |  100.00 | Using index condition |
|  1 | SIMPLE      | b    | NULL      | eq_ref | PRIMARY      | PRIMARY | 4      | test.a.id |    1 |  100.00 | NULL                  |
+—-+————-+——-+————+——–+—————+———+———+———–+——+———-+———————–+
2 rows in set, 1 warning (0.00 sec)

mysql>

可以看出在多表查询的情况下 or 条件如果不在同一个表内执行计划表 a 的查询不走索引。

我们试试看用 union all 来进行改写

mysql> explain select a.pad,a.c from sbtest1 a,sbtest2 b where a.id=b.id and a.c=’123′ union all select a.pad,a.c from sbtest1 a,sbtest2 b where a.id=b.id and b.c=’1234′;
+—-+————-+——-+————+——–+—————+———+———+———–+——+———-+————-+
| id | select_type | table | partitions | type  | possible_keys | key    | key_len | ref      | rows | filtered | Extra      |
+—-+————-+——-+————+——–+—————+———+———+———–+——+———-+————-+
|  1 | PRIMARY    | a    | NULL      | ref    | PRIMARY,c_1  | c_1    | 120    | const    |    1 |  100.00 | NULL        |
|  1 | PRIMARY    | b    | NULL      | eq_ref | PRIMARY      | PRIMARY | 4      | test.a.id |    1 |  100.00 | Using index |
|  2 | UNION      | b    | NULL      | ref    | PRIMARY,c_2  | c_2    | 120    | const    |    1 |  100.00 | Using index |
|  2 | UNION      | a    | NULL      | eq_ref | PRIMARY      | PRIMARY | 4      | test.b.id |    1 |  100.00 | NULL        |
+—-+————-+——-+————+——–+—————+———+———+———–+——+———-+————-+

在 or 的条件不在同一个表的情况下 使用 union all 来改写扫描行数减少且会走索引。

正文完
星哥玩云-微信公众号
post-qrcode
 0
星锅
版权声明:本站原创文章,由 星锅 于2022-01-22发表,共计8116字。
转载说明:除特殊说明外本站文章皆由CC-4.0协议发布,转载请注明出处。
【腾讯云】推广者专属福利,新客户无门槛领取总价值高达2860元代金券,每种代金券限量500张,先到先得。
阿里云-最新活动爆款每日限量供应
评论(没有评论)
验证码
【腾讯云】云服务器、云数据库、COS、CDN、短信等云产品特惠热卖中

星哥玩云

星哥玩云
星哥玩云
分享互联网知识
用户数
4
文章数
19348
评论数
4
阅读量
7802332
文章搜索
热门文章
开发者必备神器:阿里云 Qoder CLI 全面解析与上手指南

开发者必备神器:阿里云 Qoder CLI 全面解析与上手指南

开发者必备神器:阿里云 Qoder CLI 全面解析与上手指南 大家好,我是星哥。之前介绍了腾讯云的 Code...
星哥带你玩飞牛NAS-6:抖音视频同步工具,视频下载自动下载保存

星哥带你玩飞牛NAS-6:抖音视频同步工具,视频下载自动下载保存

星哥带你玩飞牛 NAS-6:抖音视频同步工具,视频下载自动下载保存 前言 各位玩 NAS 的朋友好,我是星哥!...
云服务器部署服务器面板1Panel:小白轻松构建Web服务与面板加固指南

云服务器部署服务器面板1Panel:小白轻松构建Web服务与面板加固指南

云服务器部署服务器面板 1Panel:小白轻松构建 Web 服务与面板加固指南 哈喽,我是星哥,经常有人问我不...
我把用了20年的360安全卫士卸载了

我把用了20年的360安全卫士卸载了

我把用了 20 年的 360 安全卫士卸载了 是的,正如标题你看到的。 原因 偷摸安装自家的软件 莫名其妙安装...
星哥带你玩飞牛NAS-3:安装飞牛NAS后的很有必要的操作

星哥带你玩飞牛NAS-3:安装飞牛NAS后的很有必要的操作

星哥带你玩飞牛 NAS-3:安装飞牛 NAS 后的很有必要的操作 前言 如果你已经有了飞牛 NAS 系统,之前...
阿里云CDN
阿里云CDN-提高用户访问的响应速度和成功率
随机文章
开源MoneyPrinterTurbo 利用AI大模型,一键生成高清短视频!

开源MoneyPrinterTurbo 利用AI大模型,一键生成高清短视频!

  开源 MoneyPrinterTurbo 利用 AI 大模型,一键生成高清短视频! 在短视频内容...
把小米云笔记搬回家:飞牛 NAS 一键部署,小米云笔记自动同步到本地

把小米云笔记搬回家:飞牛 NAS 一键部署,小米云笔记自动同步到本地

把小米云笔记搬回家:飞牛 NAS 一键部署,小米云笔记自动同步到本地 大家好,我是星哥,今天教大家在飞牛 NA...
星哥带你玩飞牛NAS-1:安装飞牛NAS

星哥带你玩飞牛NAS-1:安装飞牛NAS

星哥带你玩飞牛 NAS-1:安装飞牛 NAS 前言 在家庭和小型工作室场景中,NAS(Network Atta...
星哥带你玩飞牛NAS-5:飞牛NAS中的Docker功能介绍

星哥带你玩飞牛NAS-5:飞牛NAS中的Docker功能介绍

星哥带你玩飞牛 NAS-5:飞牛 NAS 中的 Docker 功能介绍 大家好,我是星哥,今天给大家带来如何在...
安装并使用谷歌AI编程工具Antigravity(亲测有效)

安装并使用谷歌AI编程工具Antigravity(亲测有效)

  安装并使用谷歌 AI 编程工具 Antigravity(亲测有效) 引言 Antigravity...

免费图片视频管理工具让灵感库告别混乱

一言一句话
-「
手气不错
星哥带你玩飞牛 NAS-10:备份微信聊天记录、数据到你的NAS中!

星哥带你玩飞牛 NAS-10:备份微信聊天记录、数据到你的NAS中!

星哥带你玩飞牛 NAS-10:备份微信聊天记录、数据到你的 NAS 中! 大家对「数据安全感」的需求越来越高 ...
把小米云笔记搬回家:飞牛 NAS 一键部署,小米云笔记自动同步到本地

把小米云笔记搬回家:飞牛 NAS 一键部署,小米云笔记自动同步到本地

把小米云笔记搬回家:飞牛 NAS 一键部署,小米云笔记自动同步到本地 大家好,我是星哥,今天教大家在飞牛 NA...
240 元左右!五盘位 NAS主机,7 代U硬解4K稳如狗,拓展性碾压同价位

240 元左右!五盘位 NAS主机,7 代U硬解4K稳如狗,拓展性碾压同价位

  240 元左右!五盘位 NAS 主机,7 代 U 硬解 4K 稳如狗,拓展性碾压同价位 在 NA...
开源MoneyPrinterTurbo 利用AI大模型,一键生成高清短视频!

开源MoneyPrinterTurbo 利用AI大模型,一键生成高清短视频!

  开源 MoneyPrinterTurbo 利用 AI 大模型,一键生成高清短视频! 在短视频内容...
手把手教你,购买云服务器并且安装宝塔面板

手把手教你,购买云服务器并且安装宝塔面板

手把手教你,购买云服务器并且安装宝塔面板 前言 大家好,我是星哥。星哥发现很多新手刚接触服务器时,都会被“选购...