MySQL 数据库 group by 语句怎么优化?

一、一个简单使用示例

我这里创建一张订单表

CREATETABLE`order_info`(`id`intNOTNULLAUTO_INCREMENTCOMMENT'主键',`order_no`intNOTNULLCOMMENT'订单号',`goods_id`intNOTNULLDEFAULT'0'COMMENT'商品id',`goods_name`varchar(50)NOTNULLCOMMENT'商品名称',`order_status`intNOTNULLDEFAULT'0'COMMENT'订单状态:1待支付,2成功支付,3支付失败,4已关闭',`pay_type`intNOTNULLDEFAULT'0'COMMENT'支付方式:1微信支付,2支付宝支付',`price`decimal(11,2)DEFAULTNULLCOMMENT'订单金额',`pay_time`datetimeDEFAULTNULLCOMMENT'支付时间',PRIMARYKEY(`id`))ENGINE=InnoDBDEFAULTCHARSET=utf8ROW_FORMAT=DYNAMICCOMMENT='订单信息表';

复制代码

同时也在表里插了一些数据

图片[1] - MySQL 数据库 group by 语句怎么优化? - MaxSSL

现在我们这里执行 group by 语句

selectgoods_name,count(*)asnumfromorder_infogroupbygoods_name

复制代码

图片[2] - MySQL 数据库 group by 语句怎么优化? - MaxSSL

很明显,这里就可以统计出来 每件商品一共有多少订单数据!

二、group by 原理分析

2.1、explain 分析

不同的数据库版本,用 explain 执行的结果并不一致,同样是上面 sql 语句

「MySQL 5.7 版本」

  • Extra 这个字段的Using temporary表示在执行分组的时候使用了临时表

  • Extra 这个字段的Using filesort表示使用了排序

「MySQL 8.0 版本」

我们通过对比可以发现:mysql 8.0 开始 group by 默认是没有排序的了!

接下来我们来解释下,什么是临时表。

2.2、聊一聊 Using temporary

Using temporary 表示由于排序没有走索引、使用union子查询连接查询,group_concat()count(distinct)表达式的求值等等会创建了一个内部临时表。

注意这里的临时表可能是内存上的临时表,也有可能是硬盘上的临时表,理所当然基于内存的临时表的时间消耗肯定要比基于硬盘的临时表的实际消耗小。

但不是说多大临时数据都可以直接存在内存的临时表,而是当超过最大内存临时表的最大容量就是转为存入磁盘临时表

当 mysql 需要创建临时表时,选择内存临时表还是硬盘临时表取决于参数tmp_table_sizemax_heap_table_size,当所需临时表的容量大于两者的最小值时,mysql 就会使用硬盘临时表存放数据。

用户可以在 mysql 的配置文件里修改该两个参数的值,两者的默认值均为 16M。

mysql>showglobalvariableslike'max_heap_table_size';+---------------------+----------+|Variable_name|Value|+---------------------+----------+|max_heap_table_size|16777216|+---------------------+----------+1rowinsetmysql>showglobalvariableslike'tmp_table_size';+----------------+----------+|Variable_name|Value|+----------------+----------+|tmp_table_size|16777216|+----------------+----------+1rowinset

复制代码

2.3、group by 是如何产生临时表的

同样以该 sql 分析

selectgoods_name,count(*)asnumfromorder_infogroupbygoods_name

复制代码

这个 SQL 产生临时表的执行流程如下

  1. 「创建内存临时表」,表里面有两个字段:goods_name 和 num;

  2. 全表扫描 order_info 表,取出 goods_name = 某商品(比如围巾、耳机、茶杯等)的记录

  3. 临时表没有 goods_name = 某商品的记录,直接插入,并记为 (某商品,1);

    临时表里有 goods_name = 某商品的记录,直接更新,把 num 值 +1

  4. 重复步骤 3 直至遍历完成,然后把结果集返回客户端。

这个流程的执行图如下:

图片[3] - MySQL 数据库 group by 语句怎么优化? - MaxSSL

三、group by 使用中注意的一个问题

我们来思考一个问题

select 的 列 和 group by 的 列 不一致会报错吗?

比如

selectgoods_id,goods_name,count(*)asnumfromorder_infogroupbygoods_id;

复制代码

上面我们想根据商品 id 进行分组,统计每个商品的订单数量,但是我们分组只根据 goods_id 分组,但在查询列的时候,既要返回 goods_id,也要返回 goods_name。

我们这么写因为我们知道:一样的 goods_id 一定有相同的 goods_name,所以就没必要写成 group by goods_id,goods_name;

但上面这种写法一定会被支持吗?未必!

我们分别以 mysql5.7 版本和 8.0 版本做下尝试。

mysql5.7版本

图片[4] - MySQL 数据库 group by 语句怎么优化? - MaxSSL

我们发现是可以查询的到的。

mysql8.0版本

图片[5] - MySQL 数据库 group by 语句怎么优化? - MaxSSL

我们在执行上面 sql 发现报错了,没错同样的 sql 在不同的 mysql 版本执行结果并不一样,我们看下报什么错!

出现这个错误的原因是 mysql 的 sql_mode 开启了ONLY_FULL_GROUP_BY模式

图片[6] - MySQL 数据库 group by 语句怎么优化? - MaxSSL

该模式的含义就是: 对于 group by 聚合操作,如果在 select 中的列,没有在 group by 中出现,那么这个 sql 是不合法的,因为列不在 group by 从句中。

这其实是一种更加严谨的做法。

就比如上面这个 sql,如果存在这个商品的名称被修改过了,但是它们的 id 确还是一样的,那么这个时候展示的商品名称是修改前的还是修改后的呢” />图片[7] - MySQL 数据库 group by 语句怎么优化? - MaxSSL

从执行结果来看 确实已经不存在临时表了。

五、一个很有意思的优化案例

为了让效果看去明显点,我在这里在数据库中添加了 100 万条数据(整整插了一下午呢)。

图片[8] - MySQL 数据库 group by 语句怎么优化? - MaxSSL

同时说明下当前数据库版本是8.0.22

执行得 sql 如下:

selectgoods_id,count(*)asnumfromorder_infowherepay_time>='2022-12-0100:00:00'andpay_time<='2022-12-3123:59:59'groupbygoods_id;

复制代码

5.1、不加任何索引

图片[9] - MySQL 数据库 group by 语句怎么优化? - MaxSSL

执行时间是:0.67秒

我们在执行下 explain

图片[10] - MySQL 数据库 group by 语句怎么优化? - MaxSSL

我们发现没有走任何索引,而且有临时表存在,那我是不是考虑给 goods_id 加一个索引?

5.2、仅分组字段加索引

altertableorder_infoaddindexidx_goods_id(goods_id);

复制代码

我们在执行下 explain

图片[11] - MySQL 数据库 group by 语句怎么优化? - MaxSSL

确实是走了上面创建的idx_goods_id,索引,那查询效率是不是要起飞了?

我们在执行下上面的查询 sql

图片[12] - MySQL 数据库 group by 语句怎么优化? - MaxSSL

执行时间是:21.82秒!

天啦,明明我的分组字段加了索引,而且从执行计划来看确实走了索引,而且也不存在Using temporary临时表了,怎么速度反而下来了,这是为什么呢?

原因:

虽然说我们用到了idx_goods_id索引,那我们看上图执行计划中 rows = 997982,说明啥,说明虽然走了索引,但是从扫描数据来看依然是全表扫描呢,为什么会这样?

首先 group by 用到索引,那就在索引树上索引数据,但是因为加了 where 条件,还是需要在去表里检索几乎所有的数据, 这样子,还不如直接去表里进行全表扫,这样还更快些。

所以没有索引反而更快了

5.3、查询字段和分组字段建立组合索引

那我们给 pay_time 和 goods_id 建立组合索引呢?

--先删除idx_goods_id索引dropindexidx_goods_idonorder_info;--再新建组合索引altertableorder_infoaddindexidx_payTime_goodsId(pay_time,goods_id);

我们在执行下 explain

图片[13] - MySQL 数据库 group by 语句怎么优化? - MaxSSL

这次可以很明显的看到

  • Extra 这个字段的Using index表示该查询条件确实用到了索引,而且是索引覆盖

  • Extra 这个字段的Using temporary表示在执行分组的时候使用了临时表

为什么加了索引还会有临时表存在呢,其实原因很简单

range 类型查询字段后面的索引全都无效

因为 pay_time 是范围查询,索引 goods_id 无效,所以分组一样有临时表存在!

我们在看下查询时间

图片[14] - MySQL 数据库 group by 语句怎么优化? - MaxSSL

执行时间是:0.04秒!

是不是快到起飞,虽然我们从执行计划来看依然还是存在Using temporary,但查询速度却非常快。

关键点就在Using index(索引覆盖),虽然排序是无法走索引了,但是不需要回表查询,这个效率提升是惊人的!

5.4、仅查询字段建立索引

上面说了就算建立了 pay_time,goods_id 组合索引,对于 goods_id 分组依然不走索引的。

这里我自建立 pay_time 单个索引

--先删除组合索引dropindexidx_payTime_goodsIdonorder_info;--再新建单个索引altertableorder_infoaddindexidx_pay_time(pay_time);

图片[15] - MySQL 数据库 group by 语句怎么优化? - MaxSSL

这次可以很明显的看到

  • Extra 这个字段的using index condition需要回表查询数据,但是有部分数据是在二级索引过滤后,再回表查询数据,减少了回表查询的数据行数

  • Extra 这个字段的Using MRR优化器将随机 IO 转化为顺序 IO 以降低查询过程中 IO 开销

  • Extra 这个字段的Using temporary表示在执行分组的时候使用了「临时表」

查看查询时间

图片[16] - MySQL 数据库 group by 语句怎么优化? - MaxSSL

执行时间0.56秒!

从结果看出,跟最开始不加索引查询速度相差不多,原因是什么呢?

最主要原因就是虽然走了索引,但是依然还需要回表查询,查询效率并没有提高多少!

那我们思考如何优化呢,既然上面走了回表,我们是不是可以不走回表查询,这里修改下 sql

selectgoods_id,count(*)asnumfromorder_infowhereidin(selectidfromorder_infowherepay_time>='2022-12-0100:00:00'andpay_time<='2022-12-3123:59:59')groupbygoods_id;

查看查询时间

图片[17] - MySQL 数据库 group by 语句怎么优化? - MaxSSL

执行时间0.39秒!

速度确实有提升,我们在执行下 explain

图片[18] - MySQL 数据库 group by 语句怎么优化? - MaxSSL

我们可以看到 没有了using index condition,而有了Using index,说明不需要再回表查询,而是走了索引覆盖!

本篇到这里就结束啦,希望整篇文章对你有帮助哦

© 版权声明
THE END
喜欢就支持一下吧
点赞0 分享