MySql的分组函数 ROLLUP 语法及完美解决方案
- Oracle:
GROUP BY ROLLUP(字段A) - MySQL:
GROUP BY 字段A WITH ROLLUP
解决方案1:使用 WITH ROLLUP和 GROUPING()
SELECT
CASE
WHEN GROUPING(C_JXDM) = 1 THEN 2
ELSE 0
END AS N_GROUPING,
-- GROUPING(C_JXDM) C_JXDM_GROUPING,
-- GROUPING(C_PXLB) C_PXLB_GROUPING,
COALESCE(C_JXDM, '合计') AS C_JXDM,
COALESCE(C_PXLB, '') AS C_PXLB,
COUNT(*) AS N_CNT
FROM (
-- 模拟你的数据表
SELECT 'name1' AS name, '103001' AS C_JXDM, '1' AS C_PXLB
UNION ALL
SELECT 'name2', '103001', '1'
UNION ALL
SELECT 'name3', '103001', '2'
UNION ALL
SELECT 'name4', '103001', '3'
UNION ALL
SELECT 'name5', '103001', '3'
UNION ALL
SELECT 'name6', '112001', '1'
UNION ALL
SELECT 'name7', '112001', '1'
UNION ALL
SELECT 'name8', '138001', '2'
) AS t
GROUP BY C_JXDM, C_PXLB WITH ROLLUP
HAVING GROUPING(C_PXLB) = 0 OR (GROUPING(C_JXDM) = 1 AND GROUPING(C_PXLB) = 1)
ORDER BY
GROUPING(C_JXDM) ASC,
C_JXDM,
GROUPING(C_PXLB) ASC,
C_PXLB;N_GROUPING C_JXDM C_PXLB N_CNT 0 103001 1 2 0 103001 2 1 0 103001 3 2 0 112001 1 2 0 138001 2 1 2 合计 8
解决方案2:更简洁的写法(MySQL 8.0+)
WITH sample_data AS (
SELECT 'name1' AS name, '103001' AS C_JXDM, '1' AS C_PXLB
UNION ALL SELECT 'name2', '103001', '1'
UNION ALL SELECT 'name3', '103001', '2'
UNION ALL SELECT 'name4', '103001', '3'
UNION ALL SELECT 'name5', '103001', '3'
UNION ALL SELECT 'name6', '112001', '1'
UNION ALL SELECT 'name7', '112001', '1'
UNION ALL SELECT 'name8', '138001', '2'
)
SELECT
CASE
WHEN GROUPING(C_JXDM) = 1 THEN 2
WHEN GROUPING(C_PXLB) = 0 THEN 0
ELSE 1
END AS N_GROUPING,
-- GROUPING(C_JXDM) C_JXDM_GROUPING,
-- GROUPING(C_PXLB) C_PXLB_GROUPING,
COALESCE(C_JXDM, '合计') AS C_JXDM,
COALESCE(C_PXLB, '') AS C_PXLB,
COUNT(*) AS N_CNT
FROM sample_data
GROUP BY C_JXDM, C_PXLB WITH ROLLUP
-- HAVING GROUPING(C_PXLB) = 0 OR (GROUPING(C_JXDM) = 1 AND GROUPING(C_PXLB) = 1)
ORDER BY
GROUPING(C_JXDM),
C_JXDM,
GROUPING(C_PXLB),
C_PXLB;N_GROUPING C_JXDM C_PXLB N_CNT 0 103001 1 2 0 103001 2 1 0 103001 3 2 1 103001 5 0 112001 1 2 1 112001 2 0 138001 2 1 1 138001 1 2 合计 8
加上“HAVING GROUPING(C_PXLB) = 0 OR (GROUPING(C_JXDM) = 1 AND GROUPING(C_PXLB) = 1)” 就只列出明细和总计行
解决方案3:如果只需要明细和总计(2级汇总)
WITH sample_data AS (
SELECT 'name1' AS name, '103001' AS C_JXDM, '1' AS C_PXLB
UNION ALL SELECT 'name2', '103001', '1'
UNION ALL SELECT 'name3', '103001', '2'
UNION ALL SELECT 'name4', '103001', '3'
UNION ALL SELECT 'name5', '103001', '3'
UNION ALL SELECT 'name6', '112001', '1'
UNION ALL SELECT 'name7', '112001', '1'
UNION ALL SELECT 'name8', '138001', '2'
)
SELECT
GROUPING(C_JXDM) + GROUPING(C_PXLB) AS N_GROUPING,
IFNULL(C_JXDM, '合计') AS C_JXDM,
IFNULL(C_PXLB, '') AS C_PXLB,
COUNT(*) AS N_CNT
FROM sample_data
GROUP BY C_JXDM, C_PXLB WITH ROLLUP
HAVING (GROUPING(C_JXDM) = 0 AND GROUPING(C_PXLB) = 0) -- 明细行
OR (GROUPING(C_JXDM) = 1 AND GROUPING(C_PXLB) = 1) -- 总计行
ORDER BY GROUPING(C_JXDM), C_JXDM, GROUPING(C_PXLB), C_PXLB;
N_GROUPING C_JXDM C_PXLB N_CNT
0 103001 1 2
0 103001 2 1
0 103001 3 2
0 112001 1 2
0 138001 2 1
2 合计 8解释:
- N_GROUPING 值:
0:表示明细行(既有C_JXDM分组,也有C_PXLB分组)2:表示总计行(两个维度都汇总了)
- GROUPING() 函数:
- 返回0表示该列参与分组
- 返回1表示该列是ROLLUP汇总的结果
- COALESCE/IFNULL:用于处理ROLLUP产生的NULL值
- HAVING 子句:
- 过滤掉中间的汇总行(如每个C_JXDM的小计),只保留明细和最终总计
如果你需要不同级别的汇总(如小计+总计),可以调整HAVING子句的条件。
到此这篇关于MySql的分组函数 ROLLUP 语法的文章就介绍到这了,更多相关mysql分组函数rollup语法内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!
相关文章
小白安装登录mysql-8.0.19-winx64的教程图解(新手必看)
这篇文章主要介绍了安装登录mysql-8.0.19-winx64的教程图解,非常适合新手学习参考,本文通过图文并茂的形式给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下2020-03-03
Windows 10 与 MySQL 5.5 安装使用及免安装使用详细教程(图文)
本文介绍Windows 10环境下,MySQL 5.5的安装使用及免安装使用教程,本文提供了资源下载及相关问题解决方案,非常不错,需要的朋友参考下2017-07-07
MySQL 8.0.31中使用MySQL Workbench提示配置文件错误信息解决方案
这篇文章主要介绍了MySQL 8.0.31中使用MySQL Workbench提示配置文件错误信息,本文给大家分享完美解决方案,文中补充介绍了MySQL Workbench部分出错及可能解决方案,需要的朋友可以参考下2023-01-01


最新评论