对一列求和并在第二列上减去

Sum one column and subtract over second column

我想显示两列相减。我需要从第一列中获取所有值的总和并减去第二列中的每个值。

这是 table 结构:

id | name | col1  | col2 | date
------------------------------------
432| xxx  | 0     | 15   |2015-11-17
432| yyy  | 10    | 30   |2015-11-19 
432| zzz  | 60    | 40   |2015-11-20  

433| aaa  | 0     | 60   |2015-11-17
433| bbb  | 80    | 20   |2015-11-19 
433| ccc  | 60    | 10   |2015-11-20  

公式应该是:

sum(col1) = 70  =>>>   WHERE ID = 432  

70 - col2            col3
-------------------------
=> 70 - 15         =  55
=> 70 - (30 + 15)  =  25 
=> 70 - (40 + 45)  = -15
---------------------------

sum(col1) = 140  ===>>    WHERE ID = 433  

 140 -   col2         col3
-------------------------
=> 140 -  60        =  80
=> 140 - (60 + 20)  =  60 
=> 140 - (10 + 80)  =  50

结果是 col3,输出应该是这样的

id | name | col1  | col2 | col3 |    date
-------------------------------------------
432| xxx  | 0     | 15   |  55  |  2015-11-17
432| yyy  | 10    | 30   |  25  |  2015-11-19 
432| zzz  | 60    | 40   | -15  |  2015-11-20  

433| aaa  | 0     | 60   |  80  |  2015-11-17
433| bbb  | 80    | 20   |  60  |  2015-11-19 
433| ccc  | 60    | 10   |  50  |  2015-11-20 

编辑:如果我需要的值因组而异,如 432 和 433 id 列,该怎么办。

您可以在 SELECT 字段列表中使用两个子查询。

第一个是您的 table 名为您的table 的所有行的总和,第二个是当前行之前的所有行的总和。所以你可以减去两个值。

试试这个:

SELECT T.id, T.name, T.col1, T.col2,
ISNULL(
    (SELECT SUM(T2.col1) FROM yourtable T2)
,0) - 
ISNULL(
    (SELECT SUM(T3.col2) FROM yourtable T3 
    WHERE T3.id <= T.id)
,0) as col3,
t.date
FROM yourtable T

继续Sql fiddle example

编辑

SELECT T.id, T.name, T.col1, T.col2,
ISNULL(
    (SELECT SUM(T2.col1) FROM yourtable T2 where T2.id = T.id)
,0) - 
ISNULL(
    (SELECT SUM(T3.col2) FROM yourtable T3 
    WHERE T3.id = T.id AND T3.date <= T.date)
,0) as col3,
t.date
FROM yourtable T

继续Sql Fiddle edited example

注意:深度编辑可以是不同的问题。两个查询不同

注意:ID字段不能有重复值

这应该有效:

declare @total int = (select sum(col1) from Table)
select id, name, col1, col2, @total - (select sum(col2) from Table where date <= T.date) as col3, date from Table T

我假设您想每次减去之前的总数(基于日期)。我希望这没问题。

架构信息

DECLARE @TEST TABLE
  (
     id INT,
     name VARCHAR(10),
     col1       INT,
     col2        int
  ) 

INSERT INTO @TEST VALUES 
(432,'xxx',0, 15 ),
(432,'yyy',10, 30 ),
(432,'zzz',60, 40 ),
(433,'aaa',0, 60 ),
(433,'bbb',80, 20 ),
(433,'ccc',60, 10 )

查询

    SELECT  T.id ,
        T.name ,
        T.col1 ,
        T.col2 ,
        SUM(T.col1) OVER( PARTITION BY T.id ORDER BY T.id ROWS BETWEEN UNBOUNDED PRECEDING AND UNBOUNDED FOLLOWING)
        - SUM(T.col2) OVER ( PARTITION BY T.id ORDER BY T.id ROWS BETWEEN UNBOUNDED PRECEDING AND CURRENT ROW ) AS col3
FROM    @TEST T;

结果

id | name | col1  | col2 | col3 |
---------------------------------
432  | xxx  | 0     | 15   |  55  |
432  | yyy  | 10    | 30   |  25  |
432  | zzz  | 60    | 40   | -15  |
433  | aaa  | 0     | 60   |  80  | 
433  | bbb  | 80    | 20   |  60  | 
433  | ccc  | 60    | 10   |  50  | 

SQL Fiddle

您可以使用带有交叉应用的简单select查询

    SELECT ID
    ,NAME
    ,COL1
    ,COL2
    ,A.C1 - (
        SUM(COL2) OVER (
            ORDER BY ID
            )
        ) AS COL3
FROM TABLE1 T1
CROSS APPLY (
    SELECT SUM(COL1) AS C1
    FROM TABLE1 T2
    ) A