我有一个包含三列的表,ID,Date,Value。我想对行进行排名,以便在ID中,排名随着Value至少为X的每个日期而上升,否则,排名保持不变。
给出ID,日期和值
1, 6/1, 8
1, 6/2, 12
1, 6/3, 14
1, 6/4, 9
1, 6/5, 11
我想基于至少10的值返回排名,这样我就可以拥有ID,日期,价值和排名:
1, 6/1, 8, 0
1, 6/2, 12, 1
1, 6/3, 14, 2
1, 6/4, 9, 2
1, 6/5, 11, 3
换句话说,每次超过阈值时排名都会增加,否则保持不变。
我试过的是
SELECT T1.*, X.Ranking FROM TABLE T1
LEFT JOIN ( SELECT *, DENSE_RANK( ) OVER ( PARTITION BY T2.ID ORDER BY T2.DATE ) Ranking
FROM TABLE T2 WHERE T2.VALUE >= 10 ) X
ON T1.ID = T2.ID AND T1.Date = T2.Date
这几乎可行。它让我的输出像
1, 6/1, 8, NULL
1, 6/2, 12, 1
1, 6/3, 14, 2
1, 6/4, 9, NULL
1, 6/5, 11, 3
然后,我想将第一个NULL转换为0,将第二个转换为2。
我将上述查询转换为cte并尝试了
SELECT T1.*, CASE WHEN T1.Ranking IS NULL THEN ISNULL( (
SELECT MAX( T2.Ranking )
FROM cte T2 WHERE T1.ID = T2.ID AND T1.Date > T2.Date, 0 )
ELSE T1.Ranking END NewRanking
FROM cte T1
看起来它会起作用,但我的表有200,000行,查询运行了25分钟......所以,我正在寻找比SELECT MAX更开箱即用的东西。
答案 0 :(得分:2)
您正在使用SQL Server 2012,因此您可以执行累积总和:
select t.*,
sum(case when value >= 10 then 1 else 0 end) over
(partition by id order by date) as ranking
from table t;
答案 1 :(得分:0)
编辑:这实际上不起作用。在精神上它取出之前的LAG值并递增它,但这不是LAG的工作方式......它本质上是“递归的”,导致'my_rank'是未定义的语法错误。更好的解决方案是基于累积总和的已接受答案。
<击> 如果您有SQL Server 2012(您没有标记您的问题),您可以执行以下操作:
SELECT
LAG(my_rank, 1, 0) OVER (ORDER BY DATE)
+ CASE WHEN VALUE >= 10 THEN 1 ELSE 0 END AS my_rank
FROM T1
击> <击> 撞击>