巧用SQL进行数据累加
创始人
2024-07-22 17:41:49
0

比如有这样一个需求,一张表格(User_Salary)包含每个人(UserName)每个月份(Month)发的薪水(Salary)

求这样一个结果集:每个人每月所发薪水及累计所得薪水和,如下表

UserName Month Salary
AAA 2010/12 1000
AAA 2011/01 2000
AAA 2011/02 3000
BBB 2010/12 2000
BBB 2011/01 2500
BBB 2011/02 2500

结果

UserName Month Salary Cumulation
AAA 2010/12 1000 1000
AAA 2011/01 2000 3000
AAA 2011/02 3000 6000
BBB 2010/12 2000 2000
BBB 2011/01 2500 4500
BBB 2011/02 2500 7000

当然这个结果在Excel中十分好实现,只需要一个公式就好:

注意G2的公式一定要保持***个列不动所以就是$F$2:F2,然后向下拖一下就可以,但是,这只适用于数据固定的情况下,试想,如果有100个员工的数据,岂不是要拖100下。当然也可能有其他办法,这个我就不知道了。

下面,如果用sql实现能有什么办法呢?首先想到的是游标。

对游标的确可以实现,写程序也可以实现,因为他们的思想是一样的:判断一下名字是不是已经遍历过了,如果遍历过了,就累加一下,如果没有就从0加起。这样很好理解,但是写的很费时,其实一条sql语句就可以实现的,那就是子查询。

 

  1. create table User_Salary (UserName nvarchar(200), Month nvarchar(20), Salary int)     
  2.       
  3. go     
  4.       
  5. insert into User_Salary (UserName,Month,Salary ) values('AAA','2010/12',1000)     
  6. insert into User_Salary (UserName,Month,Salary ) values('AAA','2011/01',2000)     
  7. insert into User_Salary (UserName,Month,Salary ) values('AAA','2011/02',3000)     
  8. insert into User_Salary (UserName,Month,Salary ) values('BBB','2010/12',2000)     
  9. insert into User_Salary (UserName,Month,Salary ) values('BBB','2011/01',2500)    
  10. insert into User_Salary (UserName,Month,Salary ) values('BBB','2011/02',2500)    
  11.      
  12. go    
  13.      
  14. select UserName,Month,Salary,    
  15.     Cummulation=(    
  16.         select SUM(Salary)     
  17.         from     
  18.             User_Salary i    
  19.         where     
  20.             i.UserName=o.UserName and i.Month<=o.Month    
  21.         )    
  22. from User_Salary o    
  23. order by 1,2    
  24.      
  25. go    
  26.      
  27. drop table User_Salary 

大家知道SQL查询的结果是面向集合,而这种嵌套的子查询恰恰就是在整个结果集返回之前做的对于每一行的运算。也许这样的写法不是很容易理解,那么下面这个写法应该容易理解多了。

  1. create table User_Salary (UserName nvarchar(200), Month nvarchar(20), Salary int)    
  2.       
  3. go    
  4.    
  5. insert into User_Salary (UserName,Month,Salary ) values('AAA','2010/12',1000)    
  6. insert into User_Salary (UserName,Month,Salary ) values('AAA','2011/01',2000)    
  7. insert into User_Salary (UserName,Month,Salary ) values('AAA','2011/02',3000)    
  8. insert into User_Salary (UserName,Month,Salary ) values('BBB','2010/12',2000)    
  9. insert into User_Salary (UserName,Month,Salary ) values('BBB','2011/01',2500)   
  10. insert into User_Salary (UserName,Month,Salary ) values('BBB','2011/02',2500)   
  11.     
  12. go    
  13.      
  14. select    
  15.      A.UserName,A.Month,MAX(A.Salary) as Salary,SUM (B.Salary) as Accumulation    
  16. from   
  17.      User_Salary A inner join User_Salary B   
  18.  ON    
  19.      A.UserName = B.UserName   
  20.  where    
  21.      B.Month <= A.Month   
  22.  group by   
  23.      A.UserName,A.Month   
  24. order by       
  25.      A.UserName,A.Month   
  26.      
  27. go   
  28.      
  29. drop table User_Salary 

这样用联合的方式就好理解一些,其实这样就是把每一行对应比他月份小的值分成一组,然后进行运算,如果这样不明白,那么下面的sql会使这个查询更加好理解:

  1. select     
  2.    A.*,B.*    
  3. from     
  4.    User_Salary A inner join User_Salary B    
  5. ON     
  6.    A.UserName = B.UserName     
  7. where     
  8.    B.Month <= A.Month     
  9. order by 1,2 

这样的结果就显而易见了

 

这就是子查询相关的递归(可以这么说?)算法。

 子查询执行计划

join执行计划

通过上述2个执行计划,虽然执行计划不同,但大体一致,这其中的区别我就不太明白了(不知道是先排序再join好 还是先join在排序好,但是我个人觉得第二种比较容易理解.).

原文出处:http://www.cnblogs.com/mylhei/archive/2011/03/09/1978184.html

【编辑推荐】

  1. MySQL技巧:结合相关参数 做好Limit优化
  2. SQL Server数据库六种数据移动方法
  3. MySQL数据库的优化(上)单机MySQL数据库的优化
  4. MySQL数据库的优化(下)MySQL数据库的高可用架构方案
  5. MySQL数据库安全解决方案

相关内容

热门资讯

如何允许远程连接到MySQL数... [[277004]]【51CTO.com快译】默认情况下,MySQL服务器仅侦听来自localhos...
如何利用交换机和端口设置来管理... 在网络管理中,总是有些人让管理员头疼。下面我们就将介绍一下一个网管员利用交换机以及端口设置等来进行D...
施耐德电气数据中心整体解决方案... 近日,全球能效管理专家施耐德电气正式启动大型体验活动“能效中国行——2012卡车巡展”,作为该活动的...
20个非常棒的扁平设计免费资源 Apple设备的平面图标PSD免费平板UI 平板UI套件24平图标Freen平板UI套件PSD径向平...
德国电信门户网站可实时显示全球... 德国电信周三推出一个门户网站,直观地实时提供其安装在全球各地的传感器网络检测到的网络攻击状况。该网站...
为啥国人偏爱 Mybatis,... 关于 SQL 和 ORM 的争论,永远都不会终止,我也一直在思考这个问题。昨天又跟群里的小伙伴进行...
《非诚勿扰》红人闫凤娇被曝厕所... 【51CTO.com 综合消息360安全专家提醒说,“闫凤娇”、“非诚勿扰”已经被黑客盯上成为了“木...
2012年第四季度互联网状况报... [[71653]]  北京时间4月25日消息,据国外媒体报道,全球知名的云平台公司Akamai Te...