MySql中怎么用group by?下面本篇文章给大家深入解析下group by用法,希望对大家有所帮助。
|
MySql中怎么用group by?下面本篇文章给大家深入解析下group by用法,希望对大家有所帮助。
日常开发中,我们经常会使用到
【相关推荐:mysql视频教程】 1. 使用group by的简单例子
假设用一张员工表,表结构如下: CREATE TABLE `staff` ( `id` bigint(11) NOT NULL AUTO_INCREMENT COMMENT '主键id', `id_card` varchar(20) NOT NULL COMMENT '身份证号码', `name` varchar(64) NOT NULL COMMENT '姓名', `age` int(4) NOT NULL COMMENT '年龄', `city` varchar(64) NOT NULL COMMENT '城市', PRIMARY KEY (`id`) ) ENGINE=InnoDB AUTO_INCREMENT=15 DEFAULT CHARSET=utf8 COMMENT='员工表'; 表存量的数据如下:
我们现在有这么一个需求:统计每个城市的员工数量。对应的 SQL 语句就可以这么写: select city ,count(*) as num from staff group by city; 执行结果如下:
这条SQL语句的逻辑很清楚啦,但是它的底层执行流程是怎样的呢? 2. group by 原理分析2.1 explain 分析我们先用 explain select city ,count(*) as num from staff group by city;
2.2 group by 的简单执行流程explain select city ,count(*) as num from staff group by city; 我们一起来看下这个SQL的执行流程哈
这个流程的执行图如下:
临时表的排序是怎样的呢?
对排序有兴趣深入了解的小伙伴,可以看我这篇文章哈。
3. where 和 having的区别
3.1 group by + where 的执行流程有些小伙伴觉得上一小节的SQL太简单啦,如果加了where条件之后,并且where条件列加了索引呢,执行流程是怎样? 好的,我们给它加个条件,并且加个 select city ,count(*) as num from staff where age> 30 group by city; //加索引 alter table staff add index idx_age (age); 再来expain分析一下: explain select city ,count(*) as num from staff where age> 30 group by city;
从explain 执行计划结果,可以发现查询条件命中了
执行流程如下: 1、创建内存临时表,表里有两个字段 2、扫描索引树 3、通过主键ID,回表找到city = 'X'
4、继续重复2,3步骤,找到所有满足条件的数据, 5、最后根据字段 3.2 group by + having 的执行如果你要查询每个城市的员工数量,获取到员工数量不低于3的城市,having可以很好解决你的问题,SQL酱紫写: select city ,count(*) as num from staff group by city having num >= 3; 查询结果如下:
3.3 同时有where、group by 、having的执行顺序如果一个SQL同时含有 比如这个SQL: select city ,count(*) as num from staff where age> 19 group by city having num >= 3;
3.4 where + having 区别总结
4. 使用 group by 注意的问题使用group by 主要有这几点需要注意:
4.1 group by一定要配合聚合函数使用嘛?group by 就是分组统计的意思,一般情况都是配合聚合函数
如果没有配合聚合函数使用可以吗?
比如这个SQL: select city,id_card,age from staff group by city; 查询结果是
大家对比看下,返回的就是每个分组的第一条数据
当然,平时大家使用的时候,group by还是配合聚合函数使用的,除非一些特殊场景,比如你想去重,当然去重用 4.2 group by 后面跟的字段一定要出现在select中嘛。不一定,比如以下SQL: select max(age) from staff group by city; 执行结果如下:
分组字段 4.3 到了最重要的一个注意问题啦,
这些都是导致慢SQL的x因素,我们一起来探讨优化方案哈。 5. group by的一些优化方案从哪些方向去优化呢?
我们一起来想下,执行
5.1 group by 后面的字段加索引如何保证 我们回到一下这个SQL select city ,count(*) as num from staff where age= 19 group by city; 它的执行计划
如果我们给它加个联合索引 alter table staff add index idx_age_city(age,city); 再去看执行计划,发现既不用排序,也不需要临时表啦。
加合适的索引是优化 5.2 order by null 不用排序并不是所有场景都适合加索引的,如果碰上不适合创建索引的场景,我们如何优化呢?
select city ,count(*) as num from staff group by city order by null 执行计划如下,已经没有
5.3 尽量只使用内存临时表如果 5.4 使用SQL_BIG_RESULT优化如果数据量实在太大怎么办呢?总不能无限调大 因此,如果预估数据量比较大,我们使用 示例SQl如下: select SQL_BIG_RESULT city ,count(*) as num from staff group by city; 执行计划的
执行流程如下:
6. 一个生产慢SQL如何优化最近遇到个生产慢SQL,跟group by相关的,给大家看下怎么优化哈。 表结构如下: CREATE TABLE `staff` ( `id` bigint(11) NOT NULL AUTO_INCREMENT COMMENT '主键id', `id_card` varchar(20) NOT NULL COMMENT '身份证号码', `name` varchar(64) NOT NULL COMMENT '姓名', `status` varchar(64) NOT NULL COMMENT 'Y-已激活 I-初始化 D-已删除 R-审核中', `age` int(4) NOT NULL COMMENT '年龄', `city` varchar(64) NOT NULL COMMENT '城市', `enterprise_no` varchar(64) NOT NULL COMMENT '企业号', `legal_cert_no` varchar(64) NOT NULL COMMENT '法人号码', PRIMARY KEY (`id`) ) ENGINE=InnoDB AUTO_INCREMENT=15 DEFAULT CHARSET=utf8 COMMENT='员工表'; 查询的SQL是这样的: select * from t1 where status = #{status} group by #{legal_cert_no}我们先不去探讨这个SQL的=是否合理。如果就是这么个SQL,你会怎么优化呢?有想法的小伙伴可以留言讨论哈,也可以加我微信加群探讨。如果你觉得文章那里写得不对,也可以提出来哈,一起进步,加油呀 更多编程相关知识,请访问:编程入门!! 以上就是深入了解MySql中怎么用group by?(用法详解)的详细内容,更多请关注模板之家(www.mb5.com.cn)其它相关文章! |
