【group by】mysql分组查询的案例和原理

news2024/10/5 14:00:57

【group by】mysql分组查询的案例和原理

  • 【一】group by的使用场景
  • 【二】group by的基本语法
    • 【1】基本语法
    • 【2】常用的聚合函数
      • (1)max函数:取出分组中的最大值
      • (2)avg函数:取出分组中的平均值
      • (3)count函数:统计每个分组中的数据有多少条
      • (4)sum函数:取出分组结果中的总和
      • (5)min函数:取出分组中的最小值
    • 【3】条件where和having的区别
      • (1)案例一:where过滤
      • (2)案例二:having过滤
      • (3)案例三:where+having
      • (4)案例四:按表达式或函数进行筛选
      • (5)案例五:按照多个字段分组
  • 【三】注意点和常见的报错信息
    • 【1】注意的问题
      • (1)where + having的使用
      • (2)group by一定要配合聚合函数一起使用嘛?
      • (3)group by的字段一定要出现在select中嘛?
      • (4)group by导致的慢SQL问题
    • 【2】常见的报错信息
  • 【四】group by的原理分析
    • 【1】分析案例
    • 【2】group by的原理
      • (1)explain分析
      • (2)group by 的简单执行流程
    • 【3】where和having的原理
      • (1)group by + where 的执行流程
      • (2)group by + having 的执行
      • (3)同时有where、group by 、having的执行顺序
  • 【五】group by的优化方案
    • 【1】优化的思路
    • 【2】group by 后面的字段加索引
    • 【3】order by null 不用排序
    • 【4】尽量只使用内存临时表
    • 【5】使用SQL_BIG_RESULT优化
  • 【六】实际使用案例分析

【一】group by的使用场景

【二】group by的基本语法

【1】基本语法

除了出现在group by后面的字段,如果要在select后查询其他字段,必须用聚合函数进行聚合

select 
	聚合函数,
	列(要求出现在group by的后面)
fromwhere 
	筛选条件
group by 
	分组的列表
order by 
	子句

【2】常用的聚合函数

(1)max函数:取出分组中的最大值

根据岗位来分组,并且查询每个岗位中最高工资的员工的工资是多少

select 
	max(salary),
	job_id
from 
	employees
group by 
	job_id;

(2)avg函数:取出分组中的平均值

根据部门分组,并且查询每个部门中工资的平均值

select 
	avg(salary),
	department_id
from 
	employees
where 
	email like '%a%'
group by 
	department_id

(3)count函数:统计每个分组中的数据有多少条

(4)sum函数:取出分组结果中的总和

(5)min函数:取出分组中的最小值

【3】条件where和having的区别

(1)分组查询中的筛选条件分为两类:
1-分组前筛选: 数据源是原始表,用where,放在group by前面,因为在分组前筛选
2-分组后筛选:数据源是分组后的结果集 ,用having,放在group by后面,因为在分组后进行筛选。

(1)案例一:where过滤

查询有奖金的每个领导手下最高工资的员工工资

select 
	max(salary),
	manager_id 
from 
	employees
where 
	commission_pct is not null
group by 
	manager_id

(2)案例二:having过滤

查询哪个部门的员工个数大于2

select  
	count(*),
	department_id 
from 
	employees 
group by 
	department_id 
having 
	count(*) > 2;

分组后筛选的思路:
1-查询每个部门的员工个数;
2-根据1的结果进行筛选,查询哪个部门员工个数大于2

(3)案例三:where+having

查询每个工种有奖金的员工的最高工资大于12000的工种编号和最高工资

select 
	max(salary),
	job_id
from 
	employees
where 
	commission_pct is not null
group by 
	job_id 
having 
	max(salary) > 12000;

思路:
1-首先where筛选有奖金的员工
2-然后根据岗位id分组
3-然后max(salary)取出每个分组中的工资最高值
4-最后having筛选工资最高值大于12000的组

(4)案例四:按表达式或函数进行筛选

按照员工姓名长度分组,查询每一组员工个数,筛选员工个数大于5的有哪些

select 
	count(*)  ,
	length(last_name) as len_name
from 
	employees 
group by 
	length(last_name)
having 
	count(*) > 5;

(5)案例五:按照多个字段分组

查询每个部门每个工种的员工平均工资,并且按照平均工资的高低排序

select 
	avg(salary),
	department_id,
	job_id 
from 
	employees
where 
	department_id is not null
group by 
	department_id,
	job_id
order by 
	avg(salary) desc;

【三】注意点和常见的报错信息

【1】注意的问题

(1)where + having的使用

(1)having子句用于分组后筛选,where子句用于行条件筛选
(2)having一般都是配合group by 和聚合函数一起出现如(count(),sum(),avg(),max(),min())
(3)where条件子句中不能使用聚集函数,而having子句就可以。
(4)having只能用在group by之后,where执行在group by之前

(2)group by一定要配合聚合函数一起使用嘛?

(3)group by的字段一定要出现在select中嘛?

(4)group by导致的慢SQL问题

group by使用不当,很容易就会产生慢SQL 问题。因为它既用到临时表,又默认用到排序。有时候还可能用到磁盘临时表。

【2】常见的报错信息

(1)分组后没有使用聚合函数
(2)select后的字段没有出现在group by之后

【四】group by的原理分析

【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;

执行结果如下
在这里插入图片描述

【2】group by的原理

(1)explain分析

explain select city ,count(*) as num from staff group by city;

在这里插入图片描述

(1)Extra 这个字段的Using temporary表示在执行分组的时候使用了临时表
(2)Extra 这个字段的Using filesort表示使用了排序

(2)group by 的简单执行流程

explain select city ,count(*) as num from staff group by city;

这个sql的执行流程:
(1)创建内存临时表,表里有两个字段city和num;
(2)全表扫描staff的记录,依次取出city = 'X’的记录。
1-判断临时表中是否有为 city='X’的行,没有就插入一个记录 (X,1);
2-如果临时表中有city='X’的行的行,就将x 这一行的num值加 1;
(3)遍历完成后,再根据字段city做排序,得到结果集返回给客户端。

流程的执行图如下:
在这里插入图片描述

临时表的排序流程:
就是把需要排序的字段,放到sort buffer,排完就返回。在这里注意一点哈,排序分全字段排序和rowid排序
(1)如果是全字段排序,需要查询返回的字段,都放入sort buffer,根据排序字段排完,直接返回
(2)如果是rowid排序,只是需要排序的字段放入sort buffer,然后多一次回表操作,再返回。
(3)怎么确定走的是全字段排序还是rowid 排序排序呢?由一个数据库参数控制的,max_length_for_sort_data

【3】where和having的原理

(1)group by + where 的执行流程

(1)如果加了where条件之后,并且where条件列加了索引呢,执行流程是怎样?

select city ,count(*) as num from staff where age> 30 group by city;
//加索引
alter table staff add index idx_age (age);

(2)explain分析

explain select city ,count(*) as num from staff where age> 30 group by city;

在这里插入图片描述
从explain 执行计划结果,可以发现查询条件命中了idx_age的索引,并且使用了临时表和排序。

Using index condition:表示索引下推优化,根据索引尽可能的过滤数据,然后再返回给服务器层根据where其他条件进行过滤。这里单个索引为什么会出现索引下推呢?explain出现并不代表一定是使用了索引下推,只是代表可以使用,但是不一定用了。

(3)执行流程如下
1-创建内存临时表,表里有两个字段city和num;
2-扫描索引树idx_age,找到大于年龄大于30的主键ID
3-通过主键ID,回表找到city = ‘X’
判断临时表中是否有为 city='X’的行,没有就插入一个记录 (X,1);
如果临时表中有city='X’的行的行,就将x 这一行的num值加 1;
4-继续重复2,3步骤,找到所有满足条件的数据,
5-最后根据字段city做排序,得到结果集返回给客户端。

(2)group by + having 的执行

(1)案例分析
查询每个城市的员工数量,获取到员工数量不低于3的城市,having可以很好解决

select city ,count(*) as num from staff  group by city having num >= 3;

查询结果
在这里插入图片描述
having称为分组过滤条件,它对返回的结果集操作。

(3)同时有where、group by 、having的执行顺序

(1)案例分析

select city ,count(*) as num from staff  where age> 19 group by city having num >= 3;

(2)执行流程
1-执行where子句查找符合年龄大于19的员工数据
2-group by子句对员工数据,根据城市分组。
3-对group by子句形成的城市组,运行聚集函数计算每一组的员工数量值;
4-最后用having子句选出员工数量大于等于3的城市组。

【五】group by的优化方案

【1】优化的思路

方向1:既然它默认会排序,我们不给它排是不是就行了?
方向2:既然临时表是影响group by性能的X因素,我们是不是可以不用临时表?

执行group by语句为什么需要临时表呢?group by的语义逻辑,就是统计不同的值出现的个数。如果这个这些值一开始就是有序的,我们是不是直接往下扫描统计就好了,就不用临时表来记录并统计结果了?

(1)group by 后面的字段加索引
(2)order by null 不用排序
(3)尽量只使用内存临时表
(4)使用SQL_BIG_RESULT

【2】group by 后面的字段加索引

如何保证group by后面的字段数值一开始就是有序的呢?当然就是加索引。

select city ,count(*) as num from staff where age= 19 group by city;

执行计划

在这里插入图片描述
给它加个联合索引idx_age_city(age,city)

alter table staff add index idx_age_city(age,city);

再去看执行计划,发现既不用排序,也不需要临时表
在这里插入图片描述

加合适的索引是优化group by最简单有效的优化方式。

【3】order by null 不用排序

并不是所有场景都适合加索引的,如果碰上不适合创建索引的场景,如果你的需求并不需要对结果集进行排序,可以使用order by null。

select city ,count(*) as num from staff group by city order by null

执行计划如下,已经没有filesort了
在这里插入图片描述

【4】尽量只使用内存临时表

如果group by需要统计的数据不多,我们可以尽量只使用内存临时表;因为如果group by 的过程因为内存临时表放不下数据,从而用到磁盘临时表的话,是比较耗时的。因此可以适当调大tmp_table_size参数,来避免用到磁盘临时表。

【5】使用SQL_BIG_RESULT优化

如果数据量实在太大怎么办呢?总不能无限调大tmp_table_size吧?但也不能眼睁睁看着数据先放到内存临时表,随着数据插入发现到达上限,再转成磁盘临时表吧?这样就有点不智能啦。

因此,如果预估数据量比较大,我们使用SQL_BIG_RESULT 这个提示直接用磁盘临时表。MySQl优化器发现,磁盘临时表是B+树存储,存储效率不如数组来得高。因此会直接用数组来存

示例SQl如下:

select SQL_BIG_RESULT city ,count(*) as num from staff group by city;

执行计划的Extra字段可以看到,执行没有再使用临时表,而是只有排序
在这里插入图片描述

执行流程如下:
(1)初始化 sort_buffer,放入city字段;
(2)扫描表staff,依次取出city的值,存入 sort_buffer 中;
(3)扫描完成后,对 sort_buffer的city字段做排序
(4)排序完成后,就得到了一个有序数组。
(5)根据有序数组,统计每个值出现的次数。

【六】实际使用案例分析

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/589540.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

斩获大奖!「智办事绩效」荣获钉钉年度奖项-「含钉量新锐奖」

近日,钉钉发布「含钉量年度奖」榜单,「智办事绩效」凭借与钉钉的深度融合斩获钉钉年度奖项-「含钉量新锐奖」。 ​作为钉钉优质合作伙伴,智办事绩效致力于用数字化、智能化的产品与专业的服务,解决传统绩效管理、人才培养痛点&…

【JOSEF约瑟 JDL-5200A 电流继电器 过负荷或短路启动元件 导轨安装】

名称:电流继电器;品牌:JOSEF约瑟;型号:JDL-5200A触点容量:250V2A;返回时间:≤35ms;整定范围:0.03-19.9A;特点:返回系数高、安装方便。…

前端vue面试题

四、路由 1. Vue-Router 的懒加载如何实现 非懒加载: import List from /components/list.vue const router new VueRouter({routes: [{ path: /list, component: List }] })(1)方案一(常用):使用箭头函数import动态加载 co…

content-visibility:可以提高项目的渲染性能

文章目录 一、前言二、浏览器支持程度三、CSS Containment四、跳过具有 content-visibility 的渲染工作五、关于accessibility的说明六、使用contains-intrinsic-size指定元素的自然大小七、通过content-visibility: hidden隐藏内容八、对下一幅画的交互影响(INP) 一、前言 co…

chatgpt赋能python:Python中*para:使用一个参数解决多个参数

Python中*para:使用一个参数解决多个参数 Python是广泛使用的编程语言之一,它非常灵活,可扩展性强,可以通过许多不同的方式实现编程目标。在Python中,有一种非常有用的特性,那就是使用para来代替多个参数。…

微信小程序原生开发功能合集十六:系统主页实现

本章实现小程序主页开发及对应逻辑处理,实现问卷填报相关功能,实现问卷查看、问卷查看、填报历史等功能的实现。   另外还提供小程序开发基础知识讲解课程,包括小程序开发基础知识、组件封装、常用接口组件使用及常用功能实现等内容,具体如下:    1. CSDN课程: http…

ArcGis系列-java发布GP分析结果(带样式)

1,前言 继上一篇 ArcGis系列-java调用GP分析 ,已经实现了gp工具的发布,调用,轮询,并已经分别保存好发布地图服务所需的矢量数据json,栅格数据tif文件和样式文件 要将gp分析的结果发布为图层供前端展示&am…

模版的分离编译 解决方案

回顾 对于模版,在之前我们就已经讲过,模版不支持分离编译(即声明定义不在同一文件中)。 类中,我们知道,对于代码量比较小的函数,会默认识别成内联函数,增加代码运行的效率&#xf…

【LeetCode 142】环形链表II

思路: 本题需要考虑以下两个问题: 判断链表是否环 如果有环,如何找到这个环的入口 一开始想着使用双指针的办法实现,但是发现没有什么思路,后面看了一些解析,用的快慢指针,两个指针重逢了&a…

IMX6ULL裸机篇之I2C协议

一. IIC实验简介 I2C 是最常用的通信接口,众多的传感器都会提供 I2C 接口来和主控相连。 比如摄像头、 加速度计、触摸屏等。 I.MX6U-ALPHA开发板 使用 I2C1 接口连接了一个距离传感器 AP3216C ,本章我们就来学习如何使用 I.MX6U 的 I2C 接口…

自学网络安全有什么好方法?

一、网络安全学习的误区 1.不要试图以编程为基础去学习网络安全 不要以编程为基础再开始学习网络安全,一般来说,学习编程不但学习周期长,且过渡到网络安全用到编程的用到的编程的关键点不多。一般人如果想要把编程学好再开始学习网络安全往…

深度学习AI编译器-TVM简介

1.为什么需要深度学习编译器 深度学习编译器主要为解决不同框架下训练的模型 部署到指定的某些设备上时所遇到的一系列复杂的问题,即将各种深度学习训练框架的模型 部署到各种硬件所面临的问题; 首先深度学习领域,从训练框架看,当…

C++ 编译过程(附简单实例)

C 编译的四个阶段:预处理、编译、汇编和链接。 这里以 g 为例,用到的文件分别为 mymath.h、mymath.cpp 和 main.cpp ,代码如下: // mymath.h extern int add(int, int);// mymath.cpp #include "mymath.h" // implemen…

攻防世界-web-newscenter

题目描述:打开链接,只有一个搜索框,然后就是一条一条新闻的记录 开启burpsuite,我们在搜索框中随便输入点什么,然后进行搜索,看下抓到的包 这里搜索的时候只有一个search参数,查询到的新闻数也变…

花6个月面过京东软件测试岗,拿个20K不过分吧?

背景介绍 计算机专业,代码能力一般,之前有过两段实习以及一个学校项目经历。第一份实习是大二暑期在深圳的一家互联网公司做软件测试,第二份实习由于大三暑假回国的时间比较短(小于两个月),于是找的实习是在…

通过Python的fitz库提取pdf中的图片

文章目录 前言一、fitz库是什么?二、安装fitz库三、查看fitz库版本四、pymupdf库是什么?五、安装pymupdf库六、查看pymupdf库版本七、fitz和pymupdf是什么关系?八、提取pdf中的图片1.引入库2.定义pdf路径3.打开PDF文件4.遍历所有页面5.获取页…

javaWeb ssh小区物业管理系统myeclipse开发mysql数据库MVC模式java编程计算机网页设计

一、源码特点 java ssh小区物业管理系统是一套完善的web设计系统(系统采用ssh框架进行设计开发),对理解JSP java编程开发语言有帮助,系统具有完整的源代码和数据库,系统主要采用B/S模式开发。开发环境为TOMCAT7.0…

Pytest测试框架中,setup和teardown函数的作用及使用方法

作为一名Python开发人员,您一定知道Pytest是一个广泛使用的Python测试框架。 它具有易用性和灵活性,可以帮助您快速编写测试用例,并且能够支持包括单元测试、集成测试、功能测试等多种测试类型。而其中,setup和teardown函数是Pyt…

C++三大特性—多态 “抽象类与虚函数表”

抽象类和虚函数表是 C中实现多态性的重要概念,它们对于学习 C非常重要。 掌握抽象类和虚函数表的使用方法对于理解 C的多态性是非常重要的。在 C中,通过使用抽象类和虚函数表,可以实现基于多态性的各种功能,如继承、多态、模板等。…

极简JVM结构图示

参考资料 JVM极简教程 JVM结构 JIT编译器,对于经常需要执行的字节码进行 类加载子系统 类加载器 tomcat的自定义类加载器 为了进行类的隔离,如果Tomcat直接使用AppClassLoader类加载类,那就会出现如下情况: 应用A中有个com.e…