-1

我已经使用 Row_Number() 在我的存储过程中实现分页。分页工作正常。但问题是,在实现 Row_Number() 之后,索引不起作用,即使我按部分顺序使用主键列,也会发生聚集索引 SCAN。以下是示例查询:

SELECT TOP (@insPageSize) A.RowNum, A.AdID, A.AdTitle, A.AdFor, A.AdCondition, 
A.AdExpPrice, A.CreatedDate, A.ModifiedDate, A.AdUID
FROM
(
SELECT ROW_NUMBER() OVER (ORDER BY vaa.AdID DESC) AS RowNum,
    vaa.AdID, vaa.AdTitle, vaa.CityID, vaa.AdFor, vaa.AdCondition, 
    vaa.AdExpPrice, vaa.CreatedDate, vaa.ModifiedDate, vaa.AdUID
FROM Catalogue.vwAvailableActiveAds vaa
WHERE vaa.CategoryID = @intCategoryID AND vaa.CountryCode = @chrCountryCode 
        AND vaa.CreatedDate > DATEADD(dd, -90, GETUTCDATE()) 
        AND vaa.StateID = @inbStateID AND vaa.CityID = @inbCityID
) A
WHERE A.RowNum > (@insPageSize * (@insPageNo - 1))

如果我尝试只执行内部查询:

 SELECT ROW_NUMBER() OVER (ORDER BY vaa.AdID DESC) AS RowNum,
    vaa.AdID, vaa.AdTitle, vaa.CityID, vaa.AdFor, vaa.AdCondition, 
    vaa.AdExpPrice, vaa.CreatedDate, vaa.ModifiedDate, vaa.AdUID
FROM Catalogue.vwAvailableActiveAds vaa
WHERE vaa.CategoryID = @intCategoryID AND vaa.CountryCode = @chrCountryCode 
        AND vaa.CreatedDate > DATEADD(dd, -90, GETUTCDATE()) 
        AND vaa.StateID = @inbStateID AND vaa.CityID = @inbCityID

它不使用任何索引。AdID 是主键,还有另一个非聚集索引涵盖所有 where 子句。但是会发生索引扫描。如果我从内部查询中删除 Row_Number() 并检查其执行计划,所有索引都可以正常工作,但 StateID 和 CityID 在非聚集索引中时再次显示为“谓词”。

请给我一些指导来解决我的两个问题。

4

2 回答 2

2

你期待什么,寻求?您在这里做了几件事,这使得执行查找变得非常困难:(a)返回一个范围的行;(b) 按 AdID DESC排序以获得 ROW_NUMBER() - 可能不是定义 PK 的顺序;(c) 过滤除 PK 之外的其他内容,以及 (d) 在输出中包括许多列,这些列不太可能被任何 NC 索引覆盖。很多人把手举在空中,大喊:“天哪!这是扫描!这太可怕了!” 即使在事实上,这是最有效的方法的情况下。

(仅仅因为没有发生搜索并不意味着“索引不起作用”——这只是意味着它们在满足这个查询方面可能效率更低。)

于 2013-02-23T16:28:59.263 回答
0

row_number()没有使用索引,因为必须先过滤数据。而且,这似乎需要全表扫描。或者,至少,索引的复杂组合。

如果你建立一个索引,vwAvailableActiveAds(CategoryId, COuntryCode, StateId, CityId, CreatedDate)那么它应该使用这个索引作为where子句。的应用程序row_number()仍然不会使用索引,但可能会使用更小的数据集。

顺便说一句,这假设视图实际上只是基础表上的一个选择。如果视图中的查询更复杂(即使使用where子句join、、或group by),那么该特定索引可能不是最佳方法。

于 2013-02-23T16:23:25.203 回答