SQLSERVER-使用 ROLLUP 汇总数据

表结构:

CREATE TABLE [dbo].[Students](
    [ID] [int] IDENTITY(1,1) NOT NULL,
    [StudentName] [nvarchar](50) NULL,
    [Sex] [int] NOT NULL,
    [GradeName] [nvarchar](50) NULL,
    [ClassName] [nvarchar](50) NULL,
    [BodyWeight] [decimal](18, 2) NOT NULL,
    [Area] [nvarchar](50) NULL,
 CONSTRAINT [PK_Students] PRIMARY KEY CLUSTERED
(
    [ID] ASC
)WITH (PAD_INDEX  = OFF, STATISTICS_NORECOMPUTE  = OFF, IGNORE_DUP_KEY = OFF, ALLOW_ROW_LOCKS  = ON, ALLOW_PAGE_LOCKS  = ON) ON [PRIMARY]
) ON [PRIMARY]

GO

测试数据:

INSERT [dbo].[Students] ([ID], [StudentName], [Sex], [GradeName], [ClassName], [BodyWeight], [Area]) VALUES (1, N‘张三‘, 1, N‘高一‘, N‘1班‘, CAST(140.00 AS Decimal(18, 2)), N‘中国‘)
INSERT [dbo].[Students] ([ID], [StudentName], [Sex], [GradeName], [ClassName], [BodyWeight], [Area]) VALUES (2, N‘李四‘, 1, N‘高一‘, N‘1班‘, CAST(140.00 AS Decimal(18, 2)), N‘中国‘)
INSERT [dbo].[Students] ([ID], [StudentName], [Sex], [GradeName], [ClassName], [BodyWeight], [Area]) VALUES (3, N‘王五‘, 1, N‘高一‘, N‘1班‘, CAST(155.00 AS Decimal(18, 2)), N‘中国‘)
INSERT [dbo].[Students] ([ID], [StudentName], [Sex], [GradeName], [ClassName], [BodyWeight], [Area]) VALUES (4, N‘奥巴马‘, 1, N‘高一‘, N‘2班‘, CAST(138.00 AS Decimal(18, 2)), N‘美国‘)
INSERT [dbo].[Students] ([ID], [StudentName], [Sex], [GradeName], [ClassName], [BodyWeight], [Area]) VALUES (5, N‘希拉里‘, 0, N‘高一‘, N‘2班‘, CAST(113.00 AS Decimal(18, 2)), N‘美国‘)
INSERT [dbo].[Students] ([ID], [StudentName], [Sex], [GradeName], [ClassName], [BodyWeight], [Area]) VALUES (6, N‘习XX‘, 1, N‘高一‘, N‘1班‘, CAST(110.00 AS Decimal(18, 2)), N‘中国‘)
INSERT [dbo].[Students] ([ID], [StudentName], [Sex], [GradeName], [ClassName], [BodyWeight], [Area]) VALUES (7, N‘温宝宝‘, 1, N‘高一‘, N‘1班‘, CAST(200.00 AS Decimal(18, 2)), N‘中国‘)
INSERT [dbo].[Students] ([ID], [StudentName], [Sex], [GradeName], [ClassName], [BodyWeight], [Area]) VALUES (8, N‘埃希‘, 0, N‘高一‘, N‘1班‘, CAST(123.00 AS Decimal(18, 2)), N‘澳大利亚‘)
INSERT [dbo].[Students] ([ID], [StudentName], [Sex], [GradeName], [ClassName], [BodyWeight], [Area]) VALUES (9, N‘卡特琳娜‘, 0, N‘高二‘, N‘1班‘, CAST(145.00 AS Decimal(18, 2)), N‘澳大利亚‘)
INSERT [dbo].[Students] ([ID], [StudentName], [Sex], [GradeName], [ClassName], [BodyWeight], [Area]) VALUES (10, N‘德玛西亚‘, 1, N‘高二‘, N‘2班‘, CAST(90.00 AS Decimal(18, 2)), N‘英国‘)
INSERT [dbo].[Students] ([ID], [StudentName], [Sex], [GradeName], [ClassName], [BodyWeight], [Area]) VALUES (11, N‘嘉文‘, 1, N‘高二‘, N‘2班‘, CAST(95.00 AS Decimal(18, 2)), N‘英国‘)
INSERT [dbo].[Students] ([ID], [StudentName], [Sex], [GradeName], [ClassName], [BodyWeight], [Area]) VALUES (12, N‘德邦‘, 1, N‘高二‘, N‘2班‘, CAST(102.00 AS Decimal(18, 2)), N‘英国‘)
INSERT [dbo].[Students] ([ID], [StudentName], [Sex], [GradeName], [ClassName], [BodyWeight], [Area]) VALUES (13, N‘蛮子‘, 1, N‘高三‘, N‘1班‘, CAST(160.00 AS Decimal(18, 2)), N‘刚果‘)
INSERT [dbo].[Students] ([ID], [StudentName], [Sex], [GradeName], [ClassName], [BodyWeight], [Area]) VALUES (14, N‘易大师‘, 1, N‘高三‘, N‘1班‘, CAST(120.00 AS Decimal(18, 2)), N‘刚果‘)

情况一:只有一个分类统计列,只需要一个合计。只需要增加with rollup即可。

SELECT CASE WHEN GROUPING(GradeName)=1 THEN ‘合计‘ ELSE GradeName END AS 年级 ,
      SUM(CASE WHEN Sex=1 THEN 1 ELSE 0 END) AS 男生数,
      SUM(CASE WHEN Sex=0 THEN 1 ELSE 0 END) AS 女生数,
      COUNT(Sex) AS 总数
FROM dbo.Students
GROUP BY GradeName WITH ROLLUP
ORDER BY GradeName DESC

情况二:有多个分类汇总列,只需要一个合计。增加rollup之后,需要使用GROUPING函数判断。

GROUPING函数 指示是否聚合 GROUP BY 列表中的指定列表达式。 在结果集中,如果 GROUPING 返回 1 则指示聚合;返回 0 则指示不聚合。 如果指定了 GROUP BY,则 GROUPING 只能用在 SELECT <select> 列表、HAVING 和 ORDER BY 子句中。

SELECT CASE WHEN GROUPING(GradeName)=1 THEN ‘合计‘ ELSE GradeName END AS 年级 ,
        ClassName AS 班级 ,
      SUM(CASE WHEN Sex=1 THEN 1 ELSE 0 END) AS 男生数,
      SUM(CASE WHEN Sex=0 THEN 1 ELSE 0 END) AS 女生数,
      COUNT(Sex) AS 总数
FROM dbo.Students
GROUP BY GradeName,ClassName WITH ROLLUP
HAVING GROUPING(GradeName)=1 OR GROUPING(ClassName)=0
ORDER BY GradeName DESC

SELECT CASE WHEN GROUPING(GradeName)=1 THEN ‘合计‘ ELSE GradeName END AS 年级 ,
       ClassName AS 班级 ,
       Area AS 地区 ,
      SUM(CASE WHEN Sex=1 THEN 1 ELSE 0 END) AS 男生数,
      SUM(CASE WHEN Sex=0 THEN 1 ELSE 0 END) AS 女生数,
      COUNT(Sex) AS 总数
FROM dbo.Students
GROUP BY GradeName,ClassName,Area WITH ROLLUP
HAVING GROUPING(GradeName)=1 OR (GROUPING(ClassName)=0 AND GROUPING(Area) =0)
ORDER BY GradeName DESC

情况三:有多个分类汇总列,需要显示全部的合计和小计。不需要增加判断。

SELECT CASE WHEN GROUPING(GradeName)=1 THEN ‘合计‘ ELSE GradeName END AS 年级 ,
       CASE WHEN GROUPING(GradeName)=0 AND GROUPING(ClassName)=1 THEN ‘小计‘ ELSE ClassName END AS 班级 ,
      SUM(CASE WHEN Sex=1 THEN 1 ELSE 0 END) AS 男生数,
      SUM(CASE WHEN Sex=0 THEN 1 ELSE 0 END) AS 女生数,
      COUNT(Sex) AS 总数
FROM dbo.Students
GROUP BY GradeName,ClassName WITH ROLLUP
ORDER BY GradeName DESC

情况四:有多个分类汇总列,需要显示部分的合计和小计。增加rollup之后,需要增加判断

SELECT CASE WHEN GROUPING(GradeName)=1 THEN ‘年级合计‘ ELSE GradeName END AS 年级 ,
       CASE WHEN GROUPING(GradeName)=0 AND GROUPING(ClassName)=1 THEN ‘班级小计‘ ELSE ClassName END AS 班级 ,
       CASE WHEN GROUPING(ClassName)=0 AND GROUPING(Area)=1 THEN ‘地区小计‘ ELSE Area END AS 地区 ,
      SUM(CASE WHEN Sex=1 THEN 1 ELSE 0 END) AS 男生数,
      SUM(CASE WHEN Sex=0 THEN 1 ELSE 0 END) AS 女生数,
      COUNT(Sex) AS 总数,
      GROUPING(GradeName) AS GradeName_G,
      GROUPING(ClassName) AS ClassName_G,
      GROUPING(Area) AS Area_G
FROM dbo.Students
GROUP BY GradeName,ClassName,Area WITH ROLLUP
HAVING GROUPING(GradeName)=1 OR GROUPING(Area)=0 OR GROUPING(ClassName)=0
ORDER BY GradeName DESC

参考:

使用ROLLUP 汇总数据

GROUPING

时间: 2024-10-07 15:07:24

SQLSERVER-使用 ROLLUP 汇总数据的相关文章

MySQL创建字段+数据处理函数+汇总数据(聚集函数)+分组数据

[0]README 0.1)本文部分文字描述转自"MySQL 必知必会",旨在review"MySQL创建字段+数据处理函数+汇总数据(聚集函数)+分组数据" 的基础知识: [1]创建计算字段 1)problem+solution 1.1)problem:存储在表中的数据都不是应用程序所需要的.我们需要直接从数据库中检索出转换,计算或格式化过的数据: 1.2)solution:这就是计算字段发挥作用所在了, 计算字段是运行时在 select语句内创建的: 2)字段定

一个有趣的SQL Server 层级汇总数据问题

看SQL Server大V宋大侠的博客文章,发现了一个有趣的sql server层级汇总数据问题. 具体的问题如下: parent_id emp_id emp_name total_amout     NULL 2 Andrew 200     2 1 Nancy 100     2 3 Janet 120     3 4 Michael 80     1 5 Robert 50     每个员工的总销售额=自己的销售额+其下级员工的总销售额,     比如:     Andrew = 200_

SqlServer将表中数据复制到另一张表

insert into phone2(ph,attr,type,carrier) select top 1000 ph,attr,type,carrier from phone 将表phone的字段和前1000条数据复制到Phone2表 数据库中的某个表删除重复数据(phone2表不能存在) select distinct  * into phone2 from phone 表phone的数据放到phone2中(phone2表可以存在) insert into phone2(ph,attr,ty

SQLSERVER 时间函数汇总

1.求当天的年份 (getdate(): 2012/05/08 18:07:26)    SELECT YEAR(GETDATE())     --20122. 求当天的月份       SELECT MONTH(GETDATE())   --53. 求当天的日     SELECT DAY(GETDATE())      --84. 求年月日    SELECT CONVERT(VARCHAR,GETDATE(),112)  --结果:20120508    SELECT CONVERT(VA

数据库SQL语句学习笔记(7)-汇总数据

目的:我们经常需要汇总数据而不是把它们实际检索出来,例如确定表的行数,某些列的总和,某些列的最值等.此时检索所有数据只能是浪费时间和系统资源. 第一部分:SQL给出了5个聚集函数,聚集函数是指对某些行运行的函数,计算并返回一个值. SQL聚集函数 函数 说明 AVG() 返回某列的平均值 COUNT() 返回某列的行数 MAX() 返回某列的最大值 MIN() 返回某列的最小值 SUM() 返回某列的和 1.AVG() SELECT AVG(prod_price) AS average_pric

MySQL汇总数据

汇总数据 有时对数据表的操作不是表中数据本身,而是表中数据的汇总,例如 某一列数据的平均值,最大值,最小值等.而对于这些常用的数据汇总处理,MySQL提供了函数来处理. SQL聚集函数 函数 说明 COUNT() 返回某列的行数 MAX() 返回某列最大值 MIN() 返回某列最小值 AVG() 返回某列平均值 SUM() 返回某列值之和 例子: 首先显示出products表格如下: 求出prod_price列的平均值 看起来比较怪,原表只显示了一行: 求出特定行的 如vend_id =1003

使用SQL-Server创建一个银行数据管理系统Ⅰ

使用SQL-Server创建一个银行数据管理系统Ⅰ 作者声明: 刚开始写博客,难免有些不足的地方,再就是本人初涉软件开发这一行业,是个不折不扣的小白,文章中肯定也会出现一些错误的地方,希望发现错误的朋友们可以及时的指出来,不足的地方还请各路大神们多多指教,以便本人参考和学习,多谢. 首先,要创建一个完整的数据管理系统,不是一蹴而就的,一定要要一步一步的来,不断完善,最终方能达到自己想要的结果,所以兔子在这里也是一点一点分步来做的. 创建数据库,数据库属性在这里用的是默认(不推荐使用这种偷懒的做法

解剖SQLSERVER 第二篇 对数据页面头进行逆向(译)

解剖SQLSERVER 第二篇  对数据页面头进行逆向(译) http://improve.dk/reverse-engineering-sql-server-page-headers/ 在开发OrcaMDF 的时候第一个挑战就是解析数据页面头部,我们知道数据页面分两部分,96字节的页面头部和8096字节的数据行 大神 Paul Randal 写了一篇文章很好的描述了页头结构,然而,即使文章描述得很详细,但是我还是找不出任何关于页头存储的格式 每一个字段的数据类型和他们的顺序 我们可以使用DBC

SqlServer表和excel数据批量复制方法

SqlServer表和excel数据批量复制方法 一.SqlServer表数据复制到excel方法: 1.新建查询,用sql语句把表数据读出来 2.然后,选择数据,右键"复制"(如果需要表字段名称,则点击连同标题复制) 3.在excel中直接粘贴就好了. 二.excel复制到SqlServer表数据方法1: 1.打开excel复制数据. 2.用编辑状态打开sql表 3.右键点击表最下面一行左侧的序号,选择粘贴(注意:excel的列一定要和sql表的列对应) 4.如果有自增,不要复制自增