帮助避免CURSOR进行列计算

时间:2011-05-11 23:41:25

标签: sql-server tsql sql-server-2008 cursor aggregate-functions

我在表变量中有一堆记录,如下所示:

Id     ProductId   Rank    RankCreated
1      123213      2       2011-05-02
2      123213      4       2011-05-03
3      123213      1       2011-05-03
4      155432      10      2011-05-01
5      155432      10      2011-05-02

Id 是我添加到我的表变量中的标识列(将解释为什么我需要它)。 ProductId 产品等级是表示产品在给定时间的排名的值。 RankCreated 产品排名的时间。

我想做什么:

  

计算每种产品的每个产品等级之间的“移动”。 “运动”被定义为当前 - 前一个。

所以“计算列”看起来像这样:

Id     ProductId   Rank    RankCreated   Movement
1      123213      2       2011-05-02    NULL
2      123213      4       2011-05-03    2
3      123213      1       2011-05-03    -3
4      155432      10      2011-05-01    NULL
5      155432      10      2011-05-02    0

我添加了 ID 列,以便我可以使用它来获取以前的记录。

以下是我如何将数据输入临时表:

insert into @rankhistories (productid, [rank], [rankcreated])
select a.ProductId, b.[rank]
from dbo.ProductRankHistories b
inner join dbo.Products a on a.ProductId = b.ProductId
order by a.ProductId, b.RankCreated

我真的看不出我怎么能避免光标在这里。该表变量中有6000多条记录,而我的光标解决方案需要5秒钟,这是不可接受的。

有人可以帮忙吗?

1 个答案:

答案 0 :(得分:4)

DECLARE @TV TABLE
(
Id INT IDENTITY(1,1) PRIMARY KEY,
ProductId INT,
Rank INT,   
RankCreated DATE
)


 /*Populate *6000 rows of random data*/
INSERT INTO @TV
SELECT TOP 6000 
              ROW_NUMBER() OVER (ORDER BY (SELECT 0)) / 9 AS ProductId,
              CRYPT_GEN_RANDOM(1) % 10 AS Rank,
              GETDATE() AS RankCreated
FROM master..spt_values v1,master..spt_values v2


SELECT t1.Id, 
       t1.ProductId, 
       t1.Rank, 
       t1.RankCreated, 
       t2.Rank - t1.Rank AS Movement 
FROM @TV t1
LEFT MERGE JOIN @TV t2 ON t1.Id = t2.Id+1 AND t1.ProductId=t2.ProductId
ORDER BY t1.Id