Loading [MathJax]/jax/output/CommonHTML/config.js
前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
圈层
工具
发布
首页
学习
活动
专区
圈层
工具
社区首页 >专栏 >MySQL8 窗口函数

MySQL8 窗口函数

作者头像
程序猿川子
发布于 2024-11-08 03:46:00
发布于 2024-11-08 03:46:00
16700
代码可运行
举报
运行总次数:0
代码可运行

MySQL8 还是有很多重量级变化的,一些底层优化大家在使用中有时候不易察觉,但是有一些用法,还是带给我们耳目一新的感觉,今天松哥和大家分享一下 MySQL8 里边的窗口函数。

一 什么是窗口函数

MySQL 8 中,窗口函数(Window Functions)是一类强大的分析函数,允许你在查询结果集上执行计算,而无需将数据分组到多个输出行中。窗口函数通常与 OVER() 子句一起使用,以指定数据窗口,即窗口函数将要在其上执行计算的行集。

简单来说,窗口函数的作用类似于在查询中对数据进行分组,不同的是,分组操作会把分组的结果聚合成一条记录,而窗口函数是将结果置于每一条数据记录中。

窗口函数的格式类似下面这样:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
sql 代码解读复制代码<窗口函数> OVER ([PARTITION BY <分组列> [, <分组列>...]]
                     [ORDER BY <排序列> [ASC | DESC] [, <排序列> [ASC | DESC]]...]
                     [<rows or range clause>])
  • <窗口函数> : 定义要在窗口中计算的聚合函数或其它分析函数,如 COUNTRANKSUM 等。
  • OVER : 窗口函数的核心关键字。
  • PARTITION BY : 定义要用来分组的一组列名。
  • ORDER BY : 定义用来排序的一组列名。
  • <rows or range clause> : 定义窗口的行集合。默认为 ROWS BETWEEN UNBOUNDED PRECEDING AND CURRENT ROW,表示窗口包括从窗口开始到当前行的所有行。

接下来我们通过一个实际案例来体会下窗口函数。

二 窗口函数实践

2.1 统计成绩和排名

假设我有如下一张表:

我现在想要计算学生的考试总成绩以及单科成绩排名,利用窗口函数就能快速搞定,如下:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
sql 代码解读复制代码SELECT name,subject,score,
SUM(score) OVER(PARTITION by name) AS '总分',
DENSE_RANK() OVER(PARTITION by subject ORDER BY score DESC) AS '学科排名'
from student

和窗口函数相关的就两列:

  • sum 求总分,over 中按照 name 进行分组,相当于就是计算每个人的总分。
  • dense_rank 是排序,这个函数会考虑并列的情况,但是并列并不影响排序,因为是计算每个人单科排名,所以就按照学科分组之后按照 score 排序。

最终执行结果如下:

2.2 销售统计

假设我有如下一张表:

这是一个名为 sales 的表,其中包含 id(销售记录 ID)、product_id(产品 ID)、sale_date(销售日期)和 amount(销售额)等字段。

现在有如下几个需求,大家把这几个需求搞懂了,基本上窗口函数就会用了。

计算累计销售额

需求:按产品 ID 分组,计算每个产品的累计销售额。

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
sql 代码解读复制代码SELECT 
    id, 
    product_id, 
    sale_date, 
    amount, 
    SUM(amount) OVER (PARTITION BY product_id ORDER BY sale_date) AS '累计销售额'
FROM 
    sales;

SUM(amount) OVER (PARTITION BY product_id ORDER BY sale_date) AS '累计销售额' 表示按 product_id 分组,按 sale_date 排序,计算每个产品的累计销售额。

最终查询结果如下:

计算移动平均值

需求:按产品 ID 分组,计算每个产品的最近 3 笔销售记录的移动平均销售额。

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
sql 代码解读复制代码SELECT 
    id, 
    product_id, 
    sale_date, 
    amount, 
    AVG(amount) OVER (PARTITION BY product_id ORDER BY sale_date ROWS BETWEEN 2 PRECEDING AND CURRENT ROW) AS '移动平均销售额'
FROM 
    sales;

AVG(amount) OVER (PARTITION BY product_id ORDER BY sale_date ROWS BETWEEN 2 PRECEDING AND CURRENT ROW) AS '移动平均销售额' 表示按 product_id 分组,按 sale_date 排序,计算当前行及前两行的平均销售额。

最终查询结果如下:

计算排名

需求:按产品 ID 分组,计算每个销售记录在该产品中的排名。

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
sql 代码解读复制代码SELECT 
    id, 
    product_id, 
    sale_date, 
    amount, 
    RANK() OVER (PARTITION BY product_id ORDER BY amount DESC) AS '销售金额排名'
FROM 
    sales;

RANK() OVER (PARTITION BY product_id ORDER BY amount DESC) AS '销售金额排名' 表示按 product_id 分组,按 amount 降序排序,计算每个销售记录在该产品中的排名。

最终查询结果如下:

计算百分比排名

需求:按产品 ID 分组,计算每个销售记录在该产品中的百分比排名。

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
sql 代码解读复制代码SELECT 
    id, 
    product_id, 
    sale_date, 
    amount, 
    PERCENT_RANK() OVER (PARTITION BY product_id ORDER BY amount DESC) AS '百分比排名'
FROM 
    sales;

PERCENT_RANK() OVER (PARTITION BY product_id ORDER BY amount DESC) AS '百分比排名' 表示按 product_id 分组,按 amount 降序排序,计算每个销售记录在该产品中的百分比排名。

最终查询结果如下:

计算前后行的差值

需求:按产品 ID 分组,计算每个销售记录与上一个销售记录之间的销售额差值。

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
sql 代码解读复制代码SELECT 
    id, 
    product_id, 
    sale_date, 
    amount, 
    LAG(amount, 1) OVER (PARTITION BY product_id ORDER BY sale_date) AS '上个销售记录',
    amount - LAG(amount, 1) OVER (PARTITION BY product_id ORDER BY sale_date) AS '差额'
FROM 
    sales;

LAG(amount, 1) OVER (PARTITION BY product_id ORDER BY sale_date):按 product_id 分组,按 sale_date 排序,获取当前行的上一行的 amount 值。 amount - LAG(amount, 1) OVER (PARTITION BY product_id ORDER BY sale_date):计算当前行与上一行的销售额差值。

最终查询结果如下:

计算第一个和最后一个值

需求:按产品 ID 分组,计算每个产品的第一个和最后一个销售日期。

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
sql 代码解读复制代码SELECT 
    product_id, 
    MIN(sale_date) OVER (PARTITION BY product_id) AS '第一个销售日期', 
    MAX(sale_date) OVER (PARTITION BY product_id) AS '最后一个销售日期'
FROM 
    sales;

MIN(sale_date) OVER (PARTITION BY product_id):按product_id分组,计算每个产品的第一个销售日期。 MAX(sale_date) OVER (PARTITION BY product_id):按product_id分组,计算每个产品的最后一个销售日期。

最终查询结果如下:

好啦,通过这几个小小案例,小伙伴们明白窗口函数了吧~

本文系转载,前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文系转载,前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

评论
登录后参与评论
暂无评论
推荐阅读
编辑精选文章
换一批
深入MySQL窗口函数:原理和应用
窗口函数(Window Functions)是SQL标准中的一个高级特性,它允许用户在不改变查询结果集行数的情况下,对每一行执行聚合计算或其他复杂的计算。这些计算是基于当前行与结果集中其他行之间的关系进行的。窗口函数特别适用于需要执行跨多行的计算,同时又想保持原始查询结果集的行数不变的场景。
公众号:码到三十五
2024/03/19
2.8K0
深入MySQL窗口函数:原理和应用
最强总结!数据库开窗函数完全指南!!
开窗函数(Window Functions)是SQL中强大的分析工具,允许我们在不改变结果集行数的情况下进行复杂的聚合和分析操作。本文将系统地介绍开窗函数的用法和实际应用场景。
SQL数据库开发
2024/11/12
5260
最强总结!数据库开窗函数完全指南!!
DM达梦数据库分析函数整理
在复杂的数据分析场景中,达梦数据库的分析函数扮演着至关重要的角色。它们允许用户在单个查询中对数据进行分组、排序、排名及聚合计算,极大地提升了数据分析的灵活性和效率。本篇将深入探讨达梦数据库中几种关键的分析函数,并通过具体案例SQL来解析其用法,帮助你更好地掌握这些强大的工具。
用户11147438
2024/07/02
9940
大数据学习之数据仓库代码题总结上
请编写 SQL 查询,计算从注册当天开始的每个用户在注册后第1天、第3天、第7天的学习留存率。留存率的计算方式是在注册后的特定天数内继续学习的用户数除以当天注册的用户总数。结果应包含日期、留存天数和留存率。
bxia的厨房_公众号
2024/03/05
2391
大数据学习之数据仓库代码题总结上
Oracle分析函数
和聚合函数相似,但是对于每一组记录,无论多少行,聚合函数只返回一行值,而分析函数对其中每一行记录都返回值
HUC思梦
2020/09/03
1.1K0
Oracle分析函数
用简单程序协助MySQL实现窗口函数
窗口函数是 SQL2003 标准才开始有的一系列 SQL 函数,用于应付一些复杂运算是比较方便。但是普遍使用的 MySQL 数据库对窗口函数支持得却很不好,直到最近的版本才开始有部分支持,这当然就让 MySQL 程序员很郁闷了。
星哥玩云
2022/08/17
1.4K0
用简单程序协助MySQL实现窗口函数
10道常考SQL笔试题
题目:在 Employees 表中,获取每个部门(department)薪资最高的员工的姓名、部门和工资。表结构如下:
SQL数据库开发
2024/09/24
1840
10道常考SQL笔试题
Hive常用窗口函数实战
本文介绍了Hive常见的序列函数,排名函数和窗口函数。结合业务场景展示了Hive分析函数的使用
Eights
2020/07/13
2.8K0
【数据库设计和SQL基础语法】--查询数据--聚合函数
聚合函数是一类在数据库中用于对多个行进行计算并返回单个结果的函数。它们能够对数据进行汇总、统计和计算,常用于提取有关数据集的摘要信息。聚合函数在 SQL 查询中广泛应用,包括统计总数、平均值、最大值、最小值等。
喵叔
2023/12/18
8140
MySQL8新特性窗口函数详解
MySQL8 窗口函数是一种特殊的函数,它可以在一组查询行上执行类似于聚合的操作,但是不会将查询行折叠为单个输出行,而是为每个查询行生成一个结果。窗口函数可以用来处理复杂的报表统计分析场景,例如计算移动平均值、累计和、排名等。其中博主认为它展现的主要威力在于「它能够让我们在不修改原有语句输出结果的基础上,直接添加新的聚合字段」。
wayn
2023/08/28
2950
MySQL8新特性窗口函数详解
MySQL8新特性窗口函数详解
本文博主给大家详细讲解一波 MySQL8 的新特性:「窗口函数」,相信大伙看完一定能有所收获。
wayn
2023/06/14
4680
MySQL8新特性窗口函数详解
这些SQL排名及分析函数,你知道吗?(5)
SQL的排名函数主要有ROW_NUMBER(), RANK(), 和 DENSE_RANK(),它们分别返回行号、排名和紧密排名。这三个函数的区别在于处理并列排名的方式。
万能数据的小草
2024/07/23
2530
这些SQL排名及分析函数,你知道吗?(5)
mysql中分组排序_oracle先分组后排序
​ 窗口函数(window functions),也被称为 “开窗函数”,也叫OLAP函数(Online Anallytical Processing,联机分析处理),可对数据库数据进行实时分析处理。它是数据库的标准功能之一,主流的数据库比如Oracle,PostgreSQL都支持窗口函数功能,MySQL 直到 8.0 版本才开始支持窗口函数。
全栈程序员站长
2022/11/10
8K0
MySQL窗口函数,你最熟悉的陌生人~
  这三个点虽然平时用得少,但在面试中却常被问到。值得一提的是,很多面试官对问题竟然也是一知半解。。
陈哈哈
2021/12/31
1.1K0
MySQL窗口函数,你最熟悉的陌生人~
三行五行的 SQL 只存在于教科书和培训班
教科书中 SQL 例句通常都很简单易懂,甚至可以当英语来读,这就给人造成 SQL 简单易学的印象。 但实际上,这种三行五行的 SQL 只存在于教科书和培训班,我们在现实业务中写的 SQL 不会论行,而是以 K 计的,一条 SQL 几百行 N 层嵌套,写出 3K5K 是常事,这种 SQL,完全谈不上简单易学,对专业程序员都是恶梦。 以 K 计本身倒不是大问题,需求真地复杂时,也只能写得长,Python/Java 代码可能会更长。但 SQL 的长和其它语言的长不一样,SQL 的长常常会意味着难写难懂,而且这个难写难懂和任务复杂度不成比例。除了一些最简单情况外,稍复杂些的任务,SQL 的难度就会陡增,对程序员的智商要求很高,所以经常用作应聘考题。
朱迪
2024/02/29
2780
MySQL 窗口函数详解:分析性查询的强大工具
MySQL 窗口函数提供了一种灵活的方式来处理 SQL 查询中的数据,它们允许你在不需要对数据进行分组的情况下对行集进行分析。窗口函数最常用于分析性操作,比如计算排名、累计和、移动平均值等。MySQL 从版本 8.0 开始支持窗口函数。以下是窗口函数的几个关键概念和常见用法:
科技新语
2024/12/17
1560
MySQL 窗口函数详解:分析性查询的强大工具
三行五行的 SQL 只存在于教科书和培训班
教科书中 SQL 例句通常都很简单易懂,甚至可以当英语来读,这就给人造成 SQL 简单易学的印象。 但实际上,这种三行五行的 SQL 只存在于教科书和培训班,我们在现实业务中写的 SQL 不会论行,而是以 K 计的,一条 SQL 几百行 N 层嵌套,写出 3K5K 是常事,这种 SQL,完全谈不上简单易学,对专业程序员都是恶梦。 以 K 计本身倒不是大问题,需求真地复杂时,也只能写得长,Python/Java 代码可能会更长。但 SQL 的长和其它语言的长不一样,SQL 的长常常会意味着难写难懂,而且这个难写难懂和任务复杂度不成比例。除了一些最简单情况外,稍复杂些的任务,SQL 的难度就会陡增,对程序员的智商要求很高,所以经常用作应聘考题。
朱迪
2024/11/27
710
三行五行的 SQL 只存在于教科书和培训班
告别复杂SQL:数据分析的降维打击
在大数据分析领域,窗口函数已经成为数据分析师手中不可或缺的利器。你是否遇到过需要计算销售同比环比、计算用户留存率、对数据进行分组排名这类分析需求?Apache Doris的窗口函数能够优雅地解决这些复杂的分析场景。
一臻数据
2024/12/24
1150
告别复杂SQL:数据分析的降维打击
MySQL实战面试题(附案例答案+建表语句+模拟数据+案例深度解析),练完直接碾压面试官
使用YEAR()和MONTH()函数从signup_date字段中提取年份和月份,并匹配给定的条件。
小白的大数据之旅
2024/11/20
1780
SQL 为什么动不动就 N 百行以 K 计
发明 SQL 的初衷之一显然是为了降低人们实施数据查询计算的难度。SQL 中用了不少类英语的词汇和语法,这是希望非技术人员也能掌握。确实,简单的 SQL 可以当作英语阅读,即使没有程序设计经验的人也能运用。
程序猿DD
2021/12/27
5420
SQL 为什么动不动就 N 百行以 K 计
推荐阅读
相关推荐
深入MySQL窗口函数:原理和应用
更多 >
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档
本文部分代码块支持一键运行,欢迎体验
本文部分代码块支持一键运行,欢迎体验