0

0

SQL 聚合函数与窗口函数结合应用

舞姬之光

舞姬之光

发布时间:2026-02-18 14:52:03

|

773人浏览过

|

来源于php中文网

原创

可以,但需分两层:先用group by聚合并用cte或子查询封装,外层再对聚合结果使用row_number()开窗;直接在同一select中混用会报错。

sql 聚合函数与窗口函数结合应用

GROUP BY 后还能不能用 ROW_NUMBER()?

不能直接用,会报 ERROR: column "xxx" must appear in the GROUP BY clause or be used in an aggregate function。窗口函数在 SQL 执行顺序中晚于聚合(GROUP BY → 聚合计算 → 窗口计算),所以先 GROUP BY 再想开窗,得把聚合结果当子查询或 CTE。

  • 正确做法是:先用子查询/CTE 完成聚合,外层再对聚合结果开窗
  • 别试图在同一个 SELECT 中既写 COUNT(*) 又写 ROW_NUMBER() OVER (...) 且带 GROUP BY —— 大多数数据库(PostgreSQL、MySQL 8.0+、SQL Server)都会拒绝
  • 例外:SQLite 支持部分“宽松”语法,但行为不可移植,不建议依赖

如何给每个分组内的聚合结果排序并取 Top N?

典型场景:查每个部门工资总和,并按总和降序排,只取前 3 个部门。这时需要两层逻辑:第一层算聚合,第二层开窗编号。

Ztoy网络商铺多用户版
Ztoy网络商铺多用户版

在原版的基础上做了一下修正:增加1st在线支付功能与论坛用户数据结合,vip也可与论坛相关,增加互动性vip会员的全面修正评论没有提交正文的问题特价商品的调用连接问题删掉了2个木马文件去掉了一个后门补了SQL注入补了一个过滤漏洞浮动价不能删除的问题不能够搜索问题收藏时放入购物车时出错点放入购物车弹出2个窗口修正定单不能删除问题VIP出错问题主题添加问题商家注册页导航连接问题添加了导航FLASH源文

下载
  • 用 CTE 包住聚合查询:WITH dept_sum AS (SELECT dept, SUM(salary) AS total FROM emp GROUP BY dept)
  • 外层加窗口函数:SELECT dept, total, ROW_NUMBER() OVER (ORDER BY total DESC) AS rn FROM dept_sum
  • 最后加 WHERE rn —— 注意不能在 CTE 内直接 <code>WHERE ROW_NUMBER() ,窗口函数不能出现在 WHERE 中
  • 如果要“每组内 Top N”(比如每个城市里订单数最多的 2 个用户),则窗口的 PARTITION BY city 必须和聚合维度对齐,否则逻辑错乱

SUM() OVER () 和 SUM() 的性能差多少?

差别可能极大,尤其数据量大时。SUM()聚合函数,输出一行;SUM() OVER () 是窗口函数,为每一行输出一个累计值,结果集行数不变 —— 这意味着它不会减少中间数据量,反而可能阻止优化器下推过滤条件。

  • 如果只是要全表总和,用 SUM() + GROUP BY () 或直接标量子查询,比 SUM() OVER () 更轻量
  • SUM() OVER (ORDER BY ...) 会触发排序,若 ORDER BY 字段无索引,可能产生临时文件和磁盘排序
  • PostgreSQL 中,SUM() OVER () 通常走 Seq Scan + HashAggregate,而普通 SUM() 可能被优化为 Index-Only Scan(如果有覆盖索引)
  • MySQL 8.0 对窗口函数支持较晚,某些复杂窗口帧(如 RANGE BETWEEN UNBOUNDED PRECEDING AND CURRENT ROW)性能波动明显,建议 EXPLAIN 分析执行计划

COUNT(*) OVER (PARTITION BY x) 返回的是什么?

返回当前行所在分区(即 x 值相同的全部行)的总行数,不是去重数,也不是聚合后的记录数 —— 它统计的是原始行数,哪怕这些行在 GROUP BY 后会被压成一行。

  • 常见误用:在已 GROUP BY 的查询里写 COUNT(*) OVER (PARTITION BY dept),结果不是“每个部门几条聚合记录”,而是“每个部门原始数据有几行”
  • 如果想统计每个部门聚合后的记录数,应该用 COUNT(*) 而非窗口函数,或者先 CTE 再窗口
  • 注意 NULL:如果 PARTITION BY 字段为 NULL,所有 NULL 值会被划入同一个分区(标准 SQL 行为),这点和 GROUP BY 一致
  • 在需要“显示明细+附带分组计数”的报表场景中很实用,但别把它当 GROUP BY ... HAVING COUNT > 1 的替代品
事情说清了就结束。窗口函数和聚合混用最常卡在执行顺序和作用域上,多一层子查询看着啰嗦,其实是 SQL 引擎真正能理解的表达方式。

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
数据分析工具有哪些
数据分析工具有哪些

数据分析工具有Excel、SQL、Python、R、Tableau、Power BI、SAS、SPSS和MATLAB等。详细介绍:1、Excel,具有强大的计算和数据处理功能;2、SQL,可以进行数据查询、过滤、排序、聚合等操作;3、Python,拥有丰富的数据分析库;4、R,拥有丰富的统计分析库和图形库;5、Tableau,提供了直观易用的用户界面等等。

985

2023.10.12

SQL中distinct的用法
SQL中distinct的用法

SQL中distinct的语法是“SELECT DISTINCT column1, column2,...,FROM table_name;”。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

334

2023.10.27

SQL中months_between使用方法
SQL中months_between使用方法

在SQL中,MONTHS_BETWEEN 是一个常见的函数,用于计算两个日期之间的月份差。想了解更多SQL的相关内容,可以阅读本专题下面的文章。

377

2024.02.23

SQL出现5120错误解决方法
SQL出现5120错误解决方法

SQL Server错误5120是由于没有足够的权限来访问或操作指定的数据库或文件引起的。想了解更多sql错误的相关内容,可以阅读本专题下面的文章。

1717

2024.03.06

sql procedure语法错误解决方法
sql procedure语法错误解决方法

sql procedure语法错误解决办法:1、仔细检查错误消息;2、检查语法规则;3、检查括号和引号;4、检查变量和参数;5、检查关键字和函数;6、逐步调试;7、参考文档和示例。想了解更多语法错误的相关内容,可以阅读本专题下面的文章。

373

2024.03.06

oracle数据库运行sql方法
oracle数据库运行sql方法

运行sql步骤包括:打开sql plus工具并连接到数据库。在提示符下输入sql语句。按enter键运行该语句。查看结果,错误消息或退出sql plus。想了解更多oracle数据库的相关内容,可以阅读本专题下面的文章。

1272

2024.04.07

sql中where的含义
sql中where的含义

sql中where子句用于从表中过滤数据,它基于指定条件选择特定的行。想了解更多where的相关内容,可以阅读本专题下面的文章。

585

2024.04.29

sql中删除表的语句是什么
sql中删除表的语句是什么

sql中用于删除表的语句是drop table。语法为drop table table_name;该语句将永久删除指定表的表和数据。想了解更多sql的相关内容,可以阅读本专题下面的文章。

436

2024.04.29

pixiv网页版官网登录与阅读指南_pixiv官网直达入口与在线访问方法
pixiv网页版官网登录与阅读指南_pixiv官网直达入口与在线访问方法

本专题系统整理pixiv网页版官网入口及登录访问方式,涵盖官网登录页面直达路径、在线阅读入口及快速进入方法说明,帮助用户高效找到pixiv官方网站,实现便捷、安全的网页端浏览与账号登录体验。

561

2026.02.13

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
Excel 教程
Excel 教程

共162课时 | 17.8万人学习

SciPy 教程
SciPy 教程

共10课时 | 1.6万人学习

NumPy 教程
NumPy 教程

共44课时 | 3.4万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号