sql累积和在特定条件下的空值

时间:2013-08-08 08:27:56

标签: sql sql-server sql-server-2008 null cumulative-sum

我需要创建具有以下累积总和的表:   - 如果行中没有任何值

,则value和null的总和为null

我的初始表,我想从中创建累积总和,如下所示:

+--------------------+---------------+------+-----+---------+-------+
| First_mob          | r2009         | r2010|r2011| r2012   | r2013 |
+--------------------+---------------+------+-----+---------+-------+
| 0                  | 1             | NULL |NULL | NULL    |NULL   |
| 1                  | 3             | 1    | 2   | 3       |3      |
| 2                  | 6             | 6    | 3   | NULL    |NULL   |
| 3                  | 10            | 17   | NULL| NULL    |5      |
| 4                  | 61            | 23   | NULL| 4       |NULL   | 
+--------------------+---------------+------+-----+---------+-------+

我想获得的表看起来像

+--------------------+---------------+------+-----+---------+-------+
| First_mob          | r2009         | r2010|r2011| r2012   | r2013 |
+--------------------+---------------+------+-----+---------+-------+
| 0                  | 1             | NULL |NULL | NULL    |NULL   |
| 1                  | 4             | 1    | 2   | 3       |3      |
| 2                  | 10            | 7    | 5   | 3       |3      |
| 3                  | 20            | 24   | NULL| 3       |8      |
| 4                  | 81            | 47   | NULL| 7       |NULL   | 
+--------------------+---------------+------+-----+---------+-------+

我的累计和的sql代码如下:

if OBJECT_ID('tempdb..#suma_risk_prestige_nbr') IS NOT NULL  drop table             #suma_risk_prestige_nbr
select tp1.first_mob_InDef,
       SUM(tp2.r2007) as r2007,
   SUM(tp2.r2008) as r2008,
   SUM(tp2.r2009) as r2009,
   SUM(tp2.r2010) as r2010,
   SUM(tp2.r2011) as r2011,
   SUM(tp2.r2012) as r2012,
   SUM(tp2.r2013) as r2013


into #suma_risk_prestige_nbr
from #risk_prestige_nbr tp1
inner join #risk_prestige_nbr tp2 on tp1.first_mob_InDef>=tp2.first_mob_InDef
group by tp1.first_mob_InDef,tp1.r2007,tp1.r2008,tp1.r2009,tp1.r2010,
tp1.r2011,tp1.r2012,tp1.r2013
order by tp1.first_mob_InDef,tp1.r2007,tp1.r2008,tp1.r2009,tp1.r2010,
tp1.r2011,tp1.r2012,tp1.r2013

由于

1 个答案:

答案 0 :(得分:0)

我看到您正在使用SQL Server,请阅读此链接,了解计算滚动总数 - Calculate a Running Total in SQL Server

2012年最快的方式 - 在2008年使用sum(...)(按...排序) - 使用连续row_numbers的CTE

因此,如果在最大总和之后的空值并不重要,那么你可以这样做(这不是最快的方法,但是对于快速cte,你必须创建具有序列号的表而没有间隙)。

;with cte as (
  select
     T1.First_mob,
     sum(T2.[r2009]) as [r2009],
     sum(T2.[r2010]) as [r2010],
     sum(T2.[r2011]) as [r2011],
     sum(T2.[r2012]) as [r2012],
     sum(T2.[r2013]) as [r2013]
  from Table1 as T1
     left outer join Table1 as T2 on T2.First_mob <= T1.First_mob
  group by T1.First_mob
)
select
    c1.First_mob,
    c1.[r2009] as [r2009],
    c1.[r2010] as [r2010],
    c1.[r2011] as [r2011],
    c1.[r2012] as [r2012],
    c1.[r2013] as [r2013]
from cte as c1

请参阅sql fiddle示例

更新查询有点奇怪,但这是因为我做了不透明,所以我无法在任何地方指定所有列名。也许有可能使它更有效,但现在我有了这个:

;with cte1 as (
  -- unpivoting columns to rows so we could write general queries
   select
       T1.First_mob,
       C.Name, C.Value
   from Table1 as T1
       cross apply (
           select 'r2009', [r2009] union all
           select 'r2010', [r2010] union all
           select 'r2011', [r2011] union all
           select 'r2012', [r2012] union all
           select 'r2013', [r2013]
       ) as C(Name, Value)
), cte2 as (
  -- counting running total
   select
      c1.First_mob, c1.Name, c1.Value, sum(c2.Value) as Total_Value
   from cte1 as c1
       inner join cte1 as c2 on c2.First_mob <= c1.First_mob and c2.Name = c1.Name
   group by c1.First_mob, c1.Name, c1.Value
), cte3 as (
  -- counting total sums (need later)
   select
      c1.Name, sum(c1.Value) as Value
   from cte1 as c1
   group by c1.Name
), cte4 as (
  -- removing all unnecessary values
   select
       c2.First_mob,
       c2.Name,
       case when c3.Value = c2.Total_Value and c2.Value is null then null else c2.Total_Value end as Value
    from cte2 as c2
       inner join cte3 as c3 on c3.Name = c2.Name
)
-- pivoting rows to columns
select
    c4.First_mob,
    max(case when C4.Name = 'r2009' then C4.Value end) as [r2009],
    max(case when C4.Name = 'r2010' then C4.Value end) as [r2010],
    max(case when C4.Name = 'r2011' then C4.Value end) as [r2011],
    max(case when C4.Name = 'r2012' then C4.Value end) as [r2012],
    max(case when C4.Name = 'r2013' then C4.Value end) as [r2013]
from cte4 as c4
group by c4.First_mob

请参阅sql fiddle示例