转:http://www.111cn.net/database/mssqlserver/43368.htm 本文章介绍了关于sql多级分类汇总实现方法及数据结构,有碰到问题的同学可参考一下。
据库结构如下
类别表
分类id 上级分类id 分类名称 分类级别 排序值代码如下    复制代码
id parentid categoryname categorylevel ordering
1   null      c1            1           1
2    1        c11           2           1
3    1        c12           2           2
4    1        c13           2           3
5    1        c14           2           4
6    2        c111          3           1
7    2        c112          3           2
然后 内容表是
内容id 类别id .........代码如下    复制代码
id categoryid .........
1    1       ........
2    4       ........
3    5       ........这样处理的弊端是:如果数据量大,子分类很多,达到4级以上,这方法处理极端占用数据库连接池
对性能影响很大。
如果用SQL下面的CTE递归处理的话,一次性就能把结果给查询出来,而且性能很不错
比用程序处理(数据量很大的情况),临时表性能更好,更方便 代码如下    复制代码
with area as(
select *,id px,cast(id as nvarchar(4000)) px2 from region where parentid=0
union all
select a.*,b.px,b.px2+ltrim(a.region_id) from region a join area b on a.parentid=b.id
)select * from area order by px,px2可以查询出结果—-所有分类及相应分类下子分类代码如下    复制代码
id title parentid
1 广东省 0
2 广州 1
3 白云区 2
4 深圳 1
5 湖南省 0
6 长沙 5
7 株洲 5 代码如下    复制代码with area as(
select * from region where parentid=1
union all
select a.* from region a join area b on a.parentid=b.id
)select * from area
可以查询出结果—-指定分类及相应分类下子分类
id title parentid
1 广东省 0
2 广州 1
3 白云区 2实现程序代码如下    复制代码
/*
标题:查询指定节点及其所有子节点的函数
作者:爱新觉罗.毓华(十八年风雨,守得冰山雪莲花开)
时间:2008-05-12
地点:广东深圳
*/
create table tb(id varchar(3) , pid varchar(3) , name varchar(10))
insert into tb values('001' , null  , '广东省')
insert into tb values('002' , '001' , '广州市')
insert into tb values('003' , '001' , '深圳市')
insert into tb values('004' , '002' , '天河区')
insert into tb values('005' , '003' , '罗湖区')
insert into tb values('006' , '003' , '福田区')
insert into tb values('007' , '003' , '宝安区')
insert into tb values('008' , '007' , '西乡镇')
insert into tb values('009' , '007' , '龙华镇')
insert into tb values('010' , '007' , '松岗镇')
go
--查询指定节点及其所有子节点的函数
create function f_cid(@ID varchar(3)) returns @t_level table(id varchar(3) , level int)
as
begindeclare @level intset @level = 1insert into @t_level select @id , @levelwhile @@ROWCOUNT > 0beginset @level = @level + 1insert into @t_level select a.id , @levelfrom tb a , @t_Level bwhere a.pid = b.id and b.level = @level - 1endreturn
end
go
--调用函数查询001(广东省)及其所有子节点
select a.* from tb a , f_cid('001') b where a.id = b.id order by a.id
/*
id   pid  name
---- ---- ----------
001  NULL 广东省
002  001  广州市
003  001  深圳市
004  002  天河区
005  003  罗湖区
006  003  福田区
007  003  宝安区
008  007  西乡镇
009  007  龙华镇
010  007  松岗镇
(所影响的行数为 10 行)
*/
--调用函数查询002(广州市)及其所有子节点
select a.* from tb a , f_cid('002') b where a.id = b.id order by a.id
/*
id   pid  name
---- ---- ----------
002  001  广州市
004  002  天河区
(所影响的行数为 2 行)
*/
--调用函数查询003(深圳市)及其所有子节点
select a.* from tb a , f_cid('003') b where a.id = b.id order by a.id
/*
id   pid  name
---- ---- ----------
003  001  深圳市
005  003  罗湖区
006  003  福田区
007  003  宝安区
008  007  西乡镇
009  007  龙华镇
010  007  松岗镇
(所影响的行数为 7 行)
*/
drop table tb
drop function f_cid实例2代码如下    复制代码
t1
id     parentid
m    a
n    a
e    m
f    m
x    f
y    f
z    b
t2
row    id      amount
1    a    13.00
2    b    20.00
3    e    20.00
4    f    20.00
5    x    20.00
6    y    20.00
7    z    20.00
8    e    12.00
9    x    11.00
10    f    13.00
如何得出如下结果:
row     id      amount
7    x    20.00
11    x    11.00x小计    31.00
8    y    20.00y小计    20.00
6    f    20.00
12    f    13.00f小计    84.00
5    e    20.00
10    e    12.00e小计    32.00
3    m    14.00m小计    130.00
4    n    13.00n小计    13.00
1    a    13.00a小计    156.00
9    z    20.00z小计    20.00
2    b    20.00b小计    40.00总计    196.00
实现程序
-- 示例数据 CREATE TABLE t1( id char(1), parentid char(1) ); INSERT t1 SELECT 'm', 'a' UNION ALL SELECT 'n', 'a' UNION ALL SELECT 'e', 'm' UNION ALL SELECT 'f', 'm' UNION ALL SELECT 'x', 'f' UNION ALL SELECT 'y', 'f' UNION ALL SELECT 'z', 'b'; CREATE TABLE t2( row int, id char(1), amount decimal(10, 2) ); INSERT t2 SELECT '1', 'a', '13.00' UNION ALL SELECT '2', 'b', '20.00' UNION ALL SELECT '3', 'e', '20.00' UNION ALL SELECT '4', 'f', '20.00' UNION ALL SELECT '5', 'x', '20.00' UNION ALL SELECT '6', 'y', '20.00' UNION ALL SELECT '7', 'z', '20.00' UNION ALL SELECT '8', 'e', '12.00' UNION ALL SELECT '9', 'x', '11.00' UNION ALL SELECT '10', 'f', '13.00'; GO -- 统计 -- 逐级汇总 declare @l int set @l=1 select  A.[id], [pid] = A.parentid, [sumnum] = SUM(B.amount), level=case  when exists(select * from t1 where parentid=a.[id]) then @l-1 else @l end into [#] from t1 A LEFT JOIN t2 B ON A.id = B.id GROUP BY A.id, A.parentid; if @@rowsqlserver/42852.htm target=_blank >count>0 create index IDX_#_id_pid on [#]([id],[pid]) else set @l=999 while @@rowcount>0 or @l=1 begin set @l=@l+1 update a set level=@l,[sumnum]=isnull(a.[sumnum],0)+isnull(b.[sumnum],0) from [#] a,( select aa.pid,[sumnum]=sum(aa.[sumnum]) from [#] aa,( select distinct [pid] from [#] where level=@l-1 )bb where aa.[pid]=bb.[pid] AND NOT EXISTS( SELECT * FROM [#] WHERE [PID]=aa.[PID] AND [Level]=0) GROUP BY aa.[PID] having sum(case when aa.level=0 then 1 else 0 end)=0 )b where a.[id]=b.[pid] end -- 最终结果 SELECT row = CASE WHEN GROUPING(A.row) = 0 THEN RTRIM(A.row) ELSE N'' END, id = CASE WHEN GROUPING(A.row) = 0 THEN A.id WHEN GROUPING(A.id) = 0 THEN A.id + '小计' ELSE N'总计' END, amount = CASE WHEN GROUPING(A.row) = 0 THEN SUM(A.amount) WHEN GROUPING(A.id) = 0 THEN ISNULL((SELECT SUM(B.sumnum) FROM # B WHERE A.id = B.id), SUM(A.amount)) ELSE SUM(A.amount) END FROM t2 A GROUP BY A.id, A.row WITH ROLLUP; drop table [#] GO DROP TABLE t1, t2; /*-- 结果 row          id                                     amount ------------ ----- --------------------------------------- 1            a                                       13.00 a小计                                     13.00 2            b                                       20.00 b小计                                     20.00 3            e                                       20.00 8            e                                       12.00 e小计                                     32.00 4            f                                       20.00 10           f                                       13.00 f小计                                     84.00 5            x                                       20.00 9            x                                       11.00 x小计                                     31.00 6            y                                       20.00 y小计                                     20.00 7            z                                       20.00 z小计                                     20.00 总计                                     169.00 (18 行受影响) --*/ 

转载于:https://www.cnblogs.com/lrzy/p/6278671.html

MS SQL 分类汇总参数 grouping(**)=1 rollup cubt相关推荐

  1. sql 分类汇总 列_分类汇总哪家强?R、Python、SAS、SQL?

    Excel数据透视表(Pivot Table)可以快速汇总大量数据,能够分类汇总和聚合数值数据,按类别和子类别汇总数据,功能十分强大,并且提供了展开.折叠.行列交换等交互式方法,对使用者来说十分的友好 ...

  2. sql查询语句_多字段分类汇总_多表合并

    一.最初(最简单的入门实例,单个分类汇总) 1,查询语句/*药品费用分科室汇总*/ select c.sksxx02,sum(nvl(b.nfyxx06,0))         from yygli6 ...

  3. sql月度汇总_按月度分类汇总

    按月份分类汇总 ID    订单号      日期 1     cno001    2013-6-5 2     cno001    2013-6-10 3     cno002    2013-6- ...

  4. ms office excel2013教程 - 分类汇总

    分类汇总 目标: 将学生表按学院和性别分类对于各科成绩平均分进行汇总 步骤: 先分别对学院和性别进行排序,再分别对学院和性别进行分类汇总

  5. ms sql 索引(一)

    索引的用途 索引的主要作用就是为加快检索的速度(效率).主要是为查询.这里的查询包括表自身的查询,还包括连接查询.此外,sql的查询优化器(隐式的)也依赖索引,如果使用索引比不使用索引时效率高,那优化 ...

  6. OpenShift 4 之 高可靠运行MS SQL Server 2019数据库

    <OpenShift 4.x HOL教程汇总> 文章目录 运行MS SQL Server数据库 在本地的RHEL上安装MS SQL Server客户端 为MS SQL Server的数据库 ...

  7. vivo一面翻车,整理完这份Java面经分类汇总,我突然悟了

    前言 这篇记录一下今年在vivo面试的准备所有面经,我给大家简单的做了一个分类汇总,本来是用来为vivo二面提前做准备的,谁知道一面都没过,现在人还在人才库里. 我还是很想去vivo的,希望下次能有机 ...

  8. Oracle 9i与MS SQL Server 2000之比较连载五.zz

    http://kb.cnblogs.com/a/1153156 Oracle 9i与MS SQL Server 2000之比较连载五 2008-04-14 18:13 四.Oracle中新的数据库对象 ...

  9. MS SQL 的入门学习

    SQL 语言入门教程 第一课简介 SQL 是英文Structured Query Language 的缩写,意思为结构化查询语言. SQL 语言的主要功能就是同各种数据库建立联系,进行沟通.按照ANS ...

最新文章

  1. 以实例说明如何使用C#从数据库中提取数据,按要求自动生成定制的Excel表格
  2. FSWD_BootStrap
  3. java set path_Java Path.setEffect方法代码示例
  4. 【已解决】可以访问Tomcat下webapp中的文件目录(间接实现下载功能)
  5. shell编程--流程控制for,do-while,if-then,break,continue,case等
  6. Error Based Pruning剪枝算法、代码实现与举例
  7. JavaScript学习杂记--对象和数组
  8. ArcGIS 10.2数字化线状要素时自动拼接成一条线
  9. 开发文档模板_需求文档模板一堆什么样的适合你呢?
  10. html字符串转换jsx,javascript – 将React.element转换为JSX字符串
  11. 基于神经网络的文本分类(基于Pytorch实现)
  12. VB在菜单上增加图标
  13. Windows下PowerShell监控Keepalived
  14. python复利代码_python复利
  15. pyglet: a cross-platform windowing and multimedia
  16. C++实现16进制和字符串的转换
  17. JAVA打卡记录计算时间
  18. 无人机倾斜摄影详解,三维实景建模应用
  19. 简单谈谈自己对前端的感想
  20. 青岛阳光计算机学校,青岛恒星科技学院

热门文章

  1. 【算法题】触宝2018编程题二
  2. urv中保研碰撞测试结果_深度解读中保研碰撞测试结果,多款热销车竟获“差评”...
  3. 德国战车7比1狂胜巴西
  4. pass all options of select from View to controller
  5. [Python] strip()方法
  6. 超级高铁公司Virgin Hyperloop One融资1.72亿美元 半数来自DP World
  7. Win8系统hiberfil.sys是什么文件?Win8系统hiberfil.sys怎么删除?
  8. Flash检尸倒计时,Google和微软继续加强浏览器HTML5的功能
  9. Java学习总结2---Java集合类
  10. 图解Blash数集题解