我需要从已知集合中选择每个类别的顶行(与this question有点类似)。问题是,如何使这个查询在大量的行上有效。为集合中的每个类别有效选择顶行
例如,我们创建一个表格,在几个地方存储温度记录。
CREATE TABLE #t (
placeId int,
ts datetime,
temp int,
PRIMARY KEY (ts, placeId)
)
-- insert some sample data
SET NOCOUNT ON
DECLARE @n int, @ts datetime
SELECT @n = 1000, @ts = '2000-01-01'
WHILE (@n>0) BEGIN
INSERT INTO #t VALUES (@n % 10, @ts, @n % 37)
IF (@n % 10 = 0) SET @ts = DATEADD(hour, 1, @ts)
SET @n = @n - 1
END
现在我需要获得最新的记录每个地区1,2,3
这种方式是有效的,但不能很好地(而且看上去脏)。
SELECT * FROM (
SELECT TOP 1 placeId, temp
FROM #t
WHERE placeId = 1
ORDER BY ts DESC
) t1
UNION ALL
SELECT * FROM (
SELECT TOP 1 placeId, temp
FROM #t
WHERE placeId = 2
ORDER BY ts DESC
) t2
UNION ALL
SELECT * FROM (
SELECT TOP 1 placeId, temp
FROM #t
WHERE placeId = 3
ORDER BY ts DESC
) t3
以下看起来更好但效率低得多(根据优化器,30%vs 70%)。
SELECT placeId, ts, temp FROM (
SELECT placeId, ts, temp, ROW_NUMBER() OVER (PARTITION BY placeId ORDER BY ts DESC) rownum
FROM #t
WHERE placeId IN (1, 2, 3)
) t
WHERE rownum = 1
的问题是,在#T执行聚集索引扫描和300个检索行,分类,编号,然后过滤,只留下3行后者查询执行计划中。对于前一个查询,三次获取一行。
有没有办法有效地执行查询没有大量的联合?
包含示例代码+1的问题 – 2010-06-04 15:17:59