深聊MySQL,从入门到入坟之:如何让order by、group by查询速度飞起来

本文涉及的产品
云数据库 RDS MySQL Serverless,0.5-2RCU 50GB
云数据库 RDS MySQL Serverless,价值2615元额度,1个月
简介: 深聊MySQL,从入门到入坟之:如何让order by、group by查询速度飞起来

1、引言


在结束性能知识分享,我们终于迎来了MySQL的 知识。

关于,MySQL的基本用法,小鱼也是整理的几篇文章,如:

《基本用法总括一》

《基本用法总括二》

《SQL去重查询数据》

《SQL语句多个表查询,inner join的用法》

《sql数据库中的 delete 与drop的区别》


接下来,小鱼会更详细的分享关于 MySQL的知识,包含这四大类:

①SQL优化

②MySQL索引

③MySQL锁

④事务


接下来的很长一段时间,

就让小鱼陪着你,

一起把 MySQL的姿(知)势(识) 搞起来!!!


1.1 MySQL的基本知识

按照惯例,先介绍一下 MySQL基本知识:


MySQL 是一种关联数据库管理系统,关联数据库将数据保存在不同的表中,而不是将所有数据放在一个大仓库内,这样就增加了速度并提高了灵活性。

MySQL 的 SQL “结构化查询语言”,是用于访问数据库的最常用标准化语言。

MySQL 软件采用了 GPL(GNU 通用公共许可证),由于其体积小、速度快、总体拥有成本低,尤其是开放源码这一特点,许多中小型网站为了降低网站总体拥有成本,而选择 MySQL 作为其网站数据库。


1.2 MySQL的优势

MySQL的优势有哪些,我们来瞧瞧。


・MySQL开源的,无需支付额外费用的;


・MySQL使用标准的SQL数据语言形式;


・MySQL可以运行多个系统上,并支持多个语言(如:C、C++、Python、Java、Perl、PHP、Eiffel、Ruby 等);


・MySQL可以定制,采用 GPL 协议,可修改源码来开发自己的 MySQL 系统。


所以,我们要搞好MySQL数据库,是不是非常有必要。


今天,

我们就先从 order by和group by来说起。


2、order by 原理


为了更好,更容易理解相关知识点,咱先举个例子。


也是小鱼的一贯作风, 举例说话 !


先创建一张测试表,


/* auth:carl_DJ  */
/* 使用ClassDJ这个database */
use ClassDJ;  
/* 如果表t1存在则删除表t1 */
drop table if exists t1;  
/* 创建表t1 */
CREATE TABLE 't1'(      
  'id' int(11) NOT NULL AUTO_INCREMENT,
  'a' int(20) DEFAULT NULL ,
  'b' int(20) DEFAULT NULL ,
  'c' int(20) DEFAULT NULL ,
  'd' datetime NOT NULL  DEFAULT CURRENT_TIMESTAMP ,
  PRIMARY KEY (`id`),
  KEY `idx_a_b` (`a`,`b`),
  KEY `idx_c` (`c`)
) ENGINE=InnoDB CHARSET=utf8mb4 ;
/* 如果存在存储过程insert_t1,则删除 */
drop procedure if exists insert_t1;
delimiter ;;
/* 创建存储过程insert_t1 */
create procedure insert_t1()
begin 
  /* 声明变量i */
  declare i int;  
  /* 设置i的初始值为1 */
  set i=1;    
  /* 对满足i<=10000的值进行while循环 */
  while(i<=10000)do
    /* 写入表d1中a、b两个字段,值都为i当前的值 */
    insert into t1(a,b,c) values(i,i,i); 
    /* 将i加1 */
    set i=i+1;    
  end while;
end ;;
delimiter ;
 /* 运行存储过程insert_t1 */
call insert_t1();       
/* 将id大于9000的行的a字段更新为1000 */
update t1 set a=1000 where id >9000;

创建完表,我们就来研究一下MySQL的排序原理


2.1 MySQL的排序方式

按照排序原理分,MySQL 排序方式分两种:


①通过有序索引直接返回有序数据


②通过Filesort进行的排序


那么,问题来了,


如何确定某条排序的SQL所使用的排序方式?


答:使用explain来查看该排序SQL的执行计划,重点关注****字段。


★如果Extra字段里显示是 Using index,则表示是通过有序索引直接返回有序数据


例如;


explain select id,c from t1 order by c;

image.png

★如果Extra字段里显示是 Using filesort,则表示该 SQL 是通过 Filesort 进行的排序


例如:


explain select id,d from t1 order by d;

image.png

是不是,这就一目了然 了。

Nice


2.2 Filesort 在哪里完成排序

MySQL 中的 Filesort 并不一定是在磁盘文件中进行排序的,也有可能在内存中排序,

内存排序还是磁盘排序取决于排序的数据大小和 sort_buffer_size 配置的大小。


・如果 “排序的数据大小” < sort_buffer_size: 内存排序


・如果 “排序的数据大小” > sort_buffer_size: 磁盘排序


同样,问了又来了,


如何确定使用 Filesort 排序的 SQL 是在内存还是在磁盘中进行的排序操作?


答:使用 trace 进行分析重点关注 number_of_tmp_files,


★如果等于 0,则表示排序过程没使用临时文件,在内存中就能完成排序;


★如果大于0,则表示排序过程中使用了临时文件。


举例


一、未使用临时文件排序


image.png


上图中,因为 number_of_tmp_files 等于 0,表示未使用临时文件进行排序,所以是内存排序。


参数解析:


・rows:预计扫描的行数

・examined_rows:参与排序的行

・number_of_tmp_files:使用临时文件的个数

・sort_buffer_size:sort_buffer 的大小

・sort_mode:排序模式


二、使用临时文件排序


image.png


上图中,因为 number_of_tmp_files 等于 7,所以表示使用的是磁盘排序。


number_of_tmp_files 等于 7表示:


该SQL 将需要排序的数据分为 7 份,然后每份单独排序,再存放在 7 个临时文件中,最后把 7 个临时文件合并成一个大的有序文件。


2.3 Filesort 下的排序模式

一、排序模式

Filesort 下的排序模式有三种:


・< sort_key, rowid >双路排序(又叫回表排序模式):是首先根据相应的条件取出相应的排序字段和可以直接定位行数据的行 ID,然后在 sort buffer 中进行排序,排序完后需要再次取回其它需要的字段;


・< sort_key, additional_fields >单路排序:是一次性取出满足条件行的所有字段,然后在sort buffer中进行排序;


・< sort_key, packed_additional_fields >打包数据排序模式:与单路排序相似,区别是将 char 和 varchar 字段存到 sort buffer 中时,更加紧缩。


MySQL 通过比较系统变量 max_length_for_sort_data 的大小和需要查询的字段总大小来判断使用哪种排序模式


二、判断排序模式

通过 max_length_for_sort_data 的大小和需要查询的字段总大小来判断使用哪种排序模式。


・如果 max_length_for_sort_data 比查询字段的总长度大,那么使用 < **sort_key, additional_fields >**排序模式;


・如果 max_length_for_sort_data 比查询字段的总长度小,那么使用 <sort_key, rowid> 排序模式。


三、实例举例


set session optimizer_trace="enabled=on",end_markers_in_json=on;
SET max_length_for_sort_data = 20;
/* 查询表d1的id、a、d三个字段的值,按照字段d进行排序  */
select a,d from t1 order by d; 
SELECT * FROM information_schema.OPTIMIZER_TRACE\G

OPTIMIZER_TRACE 结果中排序信息,如下图:


image.png

可以看到,使用的是**<sort_key,additional_fields>**排序模式。


我们要是变成 <sort_key, rowid>,怎么整呢?


思路:因为 a、d 两个字段的总长度为 12,可以尝试把 max_length_for_sort_data 改为小于 12 的值,看排序模式是否有改变。


操作:

set session optimizer_trace="enabled=on",end_markers_in_json=on;
SET max_length_for_sort_data = 4;
/* 查询表d1的id、a、d三个字段的值,按照字段d进行排序  */
select a,d from t1 order by d; 
SELECT * FROM information_schema.OPTIMIZER_TRACE\G

OPTIMIZER_TRACE 结果中排序信息,如下图:


image.png


看看,是不是变成 <sort_key ,rowid>排序模式了。

非常Nice。


3、order by 优化


我们了解了order by的原理,

那么我们就来看看,优化order by 有什么技巧。


3.1 添加合适索引

3.1.1 排序字段添加索引

①首先我们看下对 d 字段(没有索引)进行排序的执行计划:


explain select d,id from t1 order by d;

执行结果如下:

image.png


发现使用的是 filesort(关注 Extra 字段)。


②再看些对 c 字段(有索引)进行排序的执行计划:


explain select c,id from t1 order by c;

执行结果如下:

image.png

可以看到,根据有索引的字段排序,在 Extra 中显示的就为 Using index,表示使用的是索引排序。


注:


如果数据量比较大,显然通过有序索引直接返回有序数据效率更高


3.1.2 多个字段排序优化

问:有时面对的需求是要对多个字段进行排序,而这种情况应该怎么优化或者设计索引呢?


答:如果多个字段排序,可以在多个排序字段上添加联合索引来优化排序语句


举个例子

对 a、c 两个字段进行排序的执行计划


explain select id,a,c from t1 order by a,c;

执行结果如下:


image.png

观察 Extra 字段,发现使用的是 filesort。


再看对 a、b(a、b 两个字段有联合索引)两个字段进行排序。


explain select id,a,b from t1 order by a,b;

执行结果如下:

image.png

发现使用的是索引排序。


注:


多个字段排序的情况,如果要通过添加索引优化,得注意排序字段的顺序与联合索引中列的顺序要一致


3.1.3 先等值查询再排序的优化

问:我们更多的情况是会先根据某个字段条件查出一部分数据,然后再排序,而这类 SQL 应该如果优化呢?


答:可以通过在条件字段和排序字段添加联合索引来优化此类排序语句。


举个例子


表 d1中,根据 a=1000 过滤数据再根据 d 字段排序的执行计划如下:


explain select id,a,d from t1 where a=1000 order by d;

执行结果如下:

image.png

可以在 Extra 字段中看到 “Using filesort”,说明使用的是 filesort 排序。


再看下根据 a=1000 过滤数据在根据 b 字段排序的执行计划(a、b 两个字段有联合索引):


explain select id,a,b from t1 where a=1000 order by b;

执行结果如下:

image.png


可以在 Extra 字段中看到“Using index”,说明使用的是索引排序。


3.2 去掉不必要的返回字段

有时,我们其实并不需要查询出所有字段,但是,有的时候,我们习惯性的写出 “select * from table_name”。


举个例子


select * from t1 order by a,b;           /* 根据a和b字段排序查出所有字段的值 */

执行结果如下:

image.png


select id,a,b from t1 order by a,b;      /* 根据a和b字段排序查出id,a,b字段的值 */

执行结果如下:

image.png


通过这两个例子,我们可以看到,

查询所有字段的这条 SQL 是 filesort 排序,

而只查 id、a、b 三个字段的 SQL 是 index 排序。


那么,


为什么查询所有字段会不走索引?


是因为:


扫描整个索引并查找到没索引的行的成本比扫描全表的成本更高,所以优化器放弃使用索引。


3.3 修改参数

一、修改哪些参数?

这里就用到了在前面提到的两个参数:


max_length_for_sort_data

sort_buffer_size。


二、如何修改这两个参数的值?


・max_length_for_sort_data:

如果觉得排序效率比较低,可以适当加大 max_length_for_sort_data 的值,让优化器优先选择全字段排序。当然不能设置过大,可能会导致 CPU 利用率过低或者磁盘 I/O 过高;


・sort_buffer_size:

适当加大 sort_buffer_size 的值,尽可能让排序在内存中完成。但不能设置过大,可能导致数据库服务器 SWAP。


4、 group by 优化


默认情况,会对 group by 字段排序,

因此优化方式与 order by 基本一致,

如果目的只是分组而不用排序,可以指定 order by null 禁止排序


5、总结


今天,整理的是 order by 和group by的 一些优化技巧,

因为在写sql中,这两个还是比较常用的,

所以,记住还是有必要的。


关于order by的优化,主要就这么几点知识:


・通过添加合适索引

・去掉不必要的返回字段

・调整参数:主要是 max_length_for_sort_data 和 sort_buffer_size


回顾一下,是不是很简单,也很Nice。

嗯,细细品,慢慢品!

接下来一段时间,小鱼都会分享更多更好玩的MySQL知识。

可要坚持撸SQL哦~


相关实践学习
基于CentOS快速搭建LAMP环境
本教程介绍如何搭建LAMP环境,其中LAMP分别代表Linux、Apache、MySQL和PHP。
全面了解阿里云能为你做什么
阿里云在全球各地部署高效节能的绿色数据中心,利用清洁计算为万物互联的新世界提供源源不断的能源动力,目前开服的区域包括中国(华北、华东、华南、香港)、新加坡、美国(美东、美西)、欧洲、中东、澳大利亚、日本。目前阿里云的产品涵盖弹性计算、数据库、存储与CDN、分析与搜索、云通信、网络、管理与监控、应用服务、互联网中间件、移动服务、视频服务等。通过本课程,来了解阿里云能够为你的业务带来哪些帮助 &nbsp; &nbsp; 相关的阿里云产品:云服务器ECS 云服务器 ECS(Elastic Compute Service)是一种弹性可伸缩的计算服务,助您降低 IT 成本,提升运维效率,使您更专注于核心业务创新。产品详情: https://www.aliyun.com/product/ecs
目录
相关文章
|
7月前
|
算法 关系型数据库 MySQL
【MySQL系列】-ORDER BY……HAVING详解及limit
文章目录 创建学校联考成绩表 一、obder by…… having…… 1.格式: 2.作用: 3.使用: 4.使用细节: 二、limit 1. limit的格式: 2. limit的作用: 3.使用:
|
5月前
|
SQL 关系型数据库 MySQL
MySQL ORDER BY(排序) 语句
MySQL ORDER BY(排序) 语句
|
SQL 关系型数据库 MySQL
mysql ORDER BY自定义排序
mysql ORDER BY自定义排序
271 0
mysql ORDER BY自定义排序
|
SQL 关系型数据库 MySQL
MySQL排序ORDER BY与分页LIMIT,SQL,减少数据表的网络传输量,完整详细可收藏
MySQL排序ORDER BY与分页LIMIT,SQL,减少数据表的网络传输量,完整详细可收藏
106 0
|
关系型数据库 MySQL
朴实!简单!依你所好,MySQL排序查询ORDER BY
朴实!简单!依你所好,MySQL排序查询ORDER BY
92 0
|
SQL 关系型数据库 MySQL
MYSQL中ORDER BY(排序查询)
`[NOT] ORDER BY 字段1[ASC/DESC], 字段2[[ASC/DESC] ] ……` `ASC`表示升序,`DESC`表示降序 如果不写,默认为升序
112 0
MYSQL中ORDER BY(排序查询)
|
算法 关系型数据库 MySQL
【MySQL笔记】ORDER BY是如何工作的?
每次业务功能中总有按操作时间排序,或者按其他字段排序的需求,一想到排序我们就会想到MySQL中的ORDER BY,那在使用的过程中,我们是否会存在不合理的使用导致查询的速度下降呢?
|
关系型数据库 MySQL 数据库
【黄啊码】MySQL入门—5、数据库小技巧:单个列group by就会,多个列呢?
【黄啊码】MySQL入门—5、数据库小技巧:单个列group by就会,多个列呢?
95 0
【黄啊码】MySQL入门—5、数据库小技巧:单个列group by就会,多个列呢?
|
关系型数据库 MySQL
MySQL - Group By 1
MySQL - Group By 1
78 0
|
关系型数据库 MySQL
MySQL - Group By 有去重功能吗?
MySQL - Group By 有去重功能吗?
548 0
MySQL - Group By 有去重功能吗?