写点什么

【YashanDB 知识库】如何将 mysql 含有 group by 的 SQL 转换成崖山支持的 SQL

作者:YashanDB
  • 2024-12-13
    广东
  • 本文字数:900 字

    阅读完需:约 3 分钟

本文内容来自 YashanDB 官网,原文内容请见:https://www.yashandb.com/newsinfo/7610112.html?templateId=1718516


问题现象

以下 SQL 在 MYSQL 下均能执行成功,在崖山下执行报错。

SELECT Sname,Ssex, min(Sage) FROM Student group by Ssex;

SELECT Sname,count(0) AS counts FROM Student;

selectmin(st.Sage) ,SC.CId,min(st.SId),min(SC.score)from Student st LEFT JOIN SC ON st.SId = SC.SIdgroup by SC.CIdorder by st.Sage;
复制代码

MYSQL 执行成功:

崖山上执行报错:YAS-04316 not a single-group group function

原因

SQL-92 和更早版本不允许选择列表、HAVING 条件或 ORDER BY 列表引用未在 GROUP BY 子句中命名的非聚合列的查询。例如,此查询在标准 SQL-92 中是非法的,因为 name 选择列表中的非聚合列未出现在 GROUP BY:

SELECT Sname,Ssex, min(Sage) FROM Student group by Ssex;
复制代码


为了使查询在 SQL-92 中合法,Sname 必须从选择列表中去掉该列,或在 GROUP BY 子句中命名该列。

SQL:1999 及更高版本允许这样的非聚合列出现在选择列中。

崖山和 ORACLE 在这个点都是及 SQL-92 为准。所以在崖上执行会报错。


解决办法

1、对于非聚合列出现在选择列中,MySQL 是任取一行记录,改成用聚合函数来取最小或最大一行。

SELECT min(Sname),Ssex, min(Sage) FROM Student group by Ssex;
SELECT max(Sname),count(0) AS counts FROM Student;
复制代码

2、对于 MySQL 中 group by 加 order by,可拆分 2 层,里面一层是 group by,外面一层是 order by

SELECT*from (selectSC.CId,min(st.Sage) as mage,min(st.SId),min(SC.score)from Student st LEFT JOIN SC ON st.SId = SC.SIdgroup by SC.CId)order by mage;
复制代码

总结

SQL-92 要求非 group by 列不能单独出来在选择列中,比 SQL:1999 允许出现任选一行出现在选择列中,更严谨合理。

任选一行对于应用开发会造成疑惑,Oracle、PostgreSQL、崖山都是以 SQL-92 为准。MySQL 在 5.7 及 8.0 之后的版本

默认也是以 SQL-92 为准, SQL_MODE 默认设置为 ONLY_FULL_GROUP_BY 即 SQL-92 方式。


参考资料:

https://dev.mysql.com/doc/refman/8.4/en/group-by-handling.html

https://dev.mysql.com/doc/refman/8.4/en/group-by-modifiers.html

发布于: 刚刚阅读数: 6
用户头像

YashanDB

关注

全自研国产新型大数据管理系统 2022-02-15 加入

还未添加个人简介

评论

发布
暂无评论
【YashanDB知识库】如何将mysql含有group by的SQL转换成崖山支持的SQL_数据库_YashanDB_InfoQ写作社区