用连续指标计算
Calculate with consecutive indicator
所以这里是 table:
Name Indicator Amount
Anson 1 3.5
Anson 2 2.6
Anson 3 8.4
Anson 4 3.2
Anson 6 3.7
Ben 1 7
Ben 3 4.7
Ben 4 3.6
指标连续时如何获取总金额?这是我想要的结果:
Name Indicator Amount
Anson 1,2,3,4 17.7
Anson 6 3.7
Ben 1 7
Ben 3,4 8.3
我试过了,但它只能组合两个连续的指标。
select name_, indicator, amount, sum_amount,
(CASE WHEN diff = 1 THEN sum_amount ELSE amount END) as final_amount,
(CASE WHEN diff = 1 THEN convert(varchar(10),prev_ind)+',' +convert(varchar(10),indicator) ELSE convert(varchar(10),indicator) END) as indicator
FROM (
select name_, indicator, prev_ind, sum(indicator-prev_ind) as diff, amount,
sum(amount + prev_amount) sum_amount
from (
select name_, indicator, lag(indicator,1,0) over (order by name_, indicator) prev_ind, amount,
lag(amount,1,0) over (order by name_) prev_amount
from tabb
)g
where indicator <> 1
group by name_, indicator, prev_ind, amount
)u
谢谢!
您可以使用 lag()
/ lead()
函数解决此问题,方法是使用以下方法对 IsConsecutive 条件和名称列进行分组;
SELECT P.NAME,
LISTAGG(P.INDICATOR,',') WITHIN GROUP(ORDER BY P.INDICATOR) INDICATORS,
SUM (P.AMOUNT) TOTAL
FROM (SELECT Q.AMOUNT,
Q.NAME,
Q.INDICATOR,
CASE WHEN Q.INDICATOR +1 = Q.AFTER OR Q.INDICATOR-1 = Q.BEFORE THEN 'IsConsecutive' ELSE TO_CHAR(Q.INDICATOR) END GRP
FROM (SELECT T.AMOUNT,
T.NAME,
T.INDICATOR,
LAG (T.INDICATOR) OVER (PARTITION BY T.NAME ORDER BY T.INDICATOR) BEFORE,
LEAD (T.INDICATOR) OVER (PARTITION BY T.NAME ORDER BY T.INDICATOR) AFTER
FROM YOUR_TABLE T
)Q
)P
GROUP BY P.GRP, P.NAME
ORDER BY P.NAME
我为 Oracle 数据库编写了这个 SQL 语句。但是如果你愿意,你可以很容易地转换它。
它给出了所需的输出:
Name Indicators Total
Anson 1,2,3,4 17.7
Anson 6 3.7
Ben 1 7
Ben 3,4 8.3
Sybase 支持 list
函数。我认为获得连续值的最简单方法是从 indicator
中减去 row_number()
。连续项目的差异是恒定的:
select name, list(indicator), sum(amount)
from (select t.*,
(row_number() over (partition by name order by indicator) - indicator
) grp
from tabb t
) t
group by name, grp;
老实说,我没有看到使用 list()
真的增加了查询。您也可以只输入最小值和最大值:
select name,
cast(min(indicator) as varchar(255)) + '-' + cast(max(indicator) as varchar(255)) as range,
sum(amount)
from (select t.*, cast(indicator as varchar(255)) as inds
(row_number() over (partition by name order by indicator) - indicator
) grp
from tabb t
) t
group by name, grp;
所以这里是 table:
Name Indicator Amount
Anson 1 3.5
Anson 2 2.6
Anson 3 8.4
Anson 4 3.2
Anson 6 3.7
Ben 1 7
Ben 3 4.7
Ben 4 3.6
指标连续时如何获取总金额?这是我想要的结果:
Name Indicator Amount
Anson 1,2,3,4 17.7
Anson 6 3.7
Ben 1 7
Ben 3,4 8.3
我试过了,但它只能组合两个连续的指标。
select name_, indicator, amount, sum_amount,
(CASE WHEN diff = 1 THEN sum_amount ELSE amount END) as final_amount,
(CASE WHEN diff = 1 THEN convert(varchar(10),prev_ind)+',' +convert(varchar(10),indicator) ELSE convert(varchar(10),indicator) END) as indicator
FROM (
select name_, indicator, prev_ind, sum(indicator-prev_ind) as diff, amount,
sum(amount + prev_amount) sum_amount
from (
select name_, indicator, lag(indicator,1,0) over (order by name_, indicator) prev_ind, amount,
lag(amount,1,0) over (order by name_) prev_amount
from tabb
)g
where indicator <> 1
group by name_, indicator, prev_ind, amount
)u
谢谢!
您可以使用 lag()
/ lead()
函数解决此问题,方法是使用以下方法对 IsConsecutive 条件和名称列进行分组;
SELECT P.NAME,
LISTAGG(P.INDICATOR,',') WITHIN GROUP(ORDER BY P.INDICATOR) INDICATORS,
SUM (P.AMOUNT) TOTAL
FROM (SELECT Q.AMOUNT,
Q.NAME,
Q.INDICATOR,
CASE WHEN Q.INDICATOR +1 = Q.AFTER OR Q.INDICATOR-1 = Q.BEFORE THEN 'IsConsecutive' ELSE TO_CHAR(Q.INDICATOR) END GRP
FROM (SELECT T.AMOUNT,
T.NAME,
T.INDICATOR,
LAG (T.INDICATOR) OVER (PARTITION BY T.NAME ORDER BY T.INDICATOR) BEFORE,
LEAD (T.INDICATOR) OVER (PARTITION BY T.NAME ORDER BY T.INDICATOR) AFTER
FROM YOUR_TABLE T
)Q
)P
GROUP BY P.GRP, P.NAME
ORDER BY P.NAME
我为 Oracle 数据库编写了这个 SQL 语句。但是如果你愿意,你可以很容易地转换它。 它给出了所需的输出:
Name Indicators Total
Anson 1,2,3,4 17.7
Anson 6 3.7
Ben 1 7
Ben 3,4 8.3
Sybase 支持 list
函数。我认为获得连续值的最简单方法是从 indicator
中减去 row_number()
。连续项目的差异是恒定的:
select name, list(indicator), sum(amount)
from (select t.*,
(row_number() over (partition by name order by indicator) - indicator
) grp
from tabb t
) t
group by name, grp;
老实说,我没有看到使用 list()
真的增加了查询。您也可以只输入最小值和最大值:
select name,
cast(min(indicator) as varchar(255)) + '-' + cast(max(indicator) as varchar(255)) as range,
sum(amount)
from (select t.*, cast(indicator as varchar(255)) as inds
(row_number() over (partition by name order by indicator) - indicator
) grp
from tabb t
) t
group by name, grp;