0

0

SQL GROUP BY 聚合函数进阶应用

冰川箭仙

冰川箭仙

发布时间:2026-02-16 17:56:35

|

203人浏览过

|

来源于php中文网

原创

group by后select列必须是聚合列或group by列,否则触发error 1055;where过滤行,having过滤分组;多字段group by顺序影响索引与null处理;窗口函数可替代group by实现明细+统计。

sql group by 聚合函数进阶应用

GROUP BY 后 SELECT 列必须是聚合列或 GROUP BY 列

这是最常触发 ERROR 1055 的原因:MySQL 5.7+ 默认启用 sql_mode=ONLY_FULL_GROUP_BY,禁止在 SELECT 中引用未聚合、也未出现在 GROUP BY 中的列。

比如写 SELECT user_id, name, COUNT(*) FROM orders GROUP BY user_idname 没聚合也没分组,直接报错。

  • 正确做法:把 name 加进 GROUP BY,或用 MAX(name)/MIN(name) 显式聚合(前提是业务允许取极值)
  • 临时绕过:不推荐,但可执行 SET sql_mode=(SELECT REPLACE(@@sql_mode,'ONLY_FULL_GROUP_BY','')),仅用于调试
  • 注意:PostgreSQL 和 SQL Server 更严格,连 MAX(name) 都要求语义明确;SQLite 则默认宽松,容易掩盖逻辑问题

HAVING 和 WHERE 的分工不能混用

WHERE 过滤行,HAVING 过滤分组——这个边界一旦模糊,结果就容易错得悄无声息。

例如想查“订单数超 5 的用户”,写成 WHERE COUNT(*) > 5 会报错,因为 COUNT(*)WHERE 阶段还没计算出来。

  • WHERE 放原始字段条件:WHERE status = 'paid'(过滤单条订单)
  • HAVING 放聚合结果条件:HAVING COUNT(*) > 5(过滤分组后统计值)
  • 性能影响:先 WHEREGROUP BY 能显著减少分组数据量;反过来把条件塞进 HAVING,等于全表分组完才筛,慢且浪费

多字段 GROUP BY 的顺序和 NULL 处理

GROUP BY a, b 不等于 GROUP BY b, a,虽然结果行数一样,但分组键的组合顺序会影响索引利用和排序行为。

Kive
Kive

一站式AI图像生成和管理平台

下载

更隐蔽的是 NULL:在大多数数据库中,NULL = NULL 为 false,但 GROUP BY 把所有 NULL 归为同一组——这和 ORDER BYNULLS FIRST/LAST 的逻辑不一致,容易误判。

  • 建索引时优先按 GROUP BY 字段顺序创建复合索引,如 INDEX idx_user_status (user_id, status)GROUP BY user_id, status 有效
  • 若需区分 NULL 和空字符串,提前用 COALESCE(status, '_null_') 转换,避免分组合并
  • PostgreSQL 支持 GROUP BY a IS NOT NULL, a 实现 NULL 单独成组,MySQL 不支持,得用 CASE WHEN 模拟

窗口函数替代 GROUP BY 的场景

当既要分组统计,又要保留原始明细行(比如每笔订单显示“该用户总消费额”),硬套 GROUP BY + JOIN 不仅啰嗦,还可能因多对一引发重复或丢失。

这时直接上窗口函数更干净:SUM(amount) OVER (PARTITION BY user_id) 就能原地算出每人总额,不改变行数。

  • 兼容性:MySQL 8.0+、PostgreSQL 9.4+、SQL Server 2012+ 支持;老版本 MySQL 只能靠变量或自连接模拟,稳定性差
  • 性能:窗口函数通常比 GROUP BY + JOIN 快,尤其大数据量下,避免了中间结果物化
  • 注意:PARTITION BY 语义等价于 GROUP BY 分组,但不能在同一个查询里同时用 GROUP BY 和窗口函数聚合同一字段,会报错

GROUP BY 看似简单,真正卡住人的往往是隐式行为:NULL 怎么分组、执行顺序怎么影响结果、不同数据库对标准的松紧程度。这些细节不跑真实数据很难暴露,光看文档容易漏掉。

热门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,提供了直观易用的用户界面等等。

963

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的相关内容,可以阅读本专题下面的文章。

375

2024.02.23

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

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

1676

2024.03.06

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

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

370

2024.03.06

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

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

1232

2024.04.07

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

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

583

2024.04.29

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

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

435

2024.04.29

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

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

283

2026.02.13

热门下载

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

精品课程

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

共162课时 | 17.5万人学习

SciPy 教程
SciPy 教程

共10课时 | 1.6万人学习

NumPy 教程
NumPy 教程

共44课时 | 3.3万人学习

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

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