示例模式:为什么优化器不使用我独特的过滤索引?
create table dbo.Person (
ID int identity(1,1) not null
constraint PK_Person primary key,
UserName nvarchar(50) null,
EncryptedPassword nvarchar(100) null
)
create index IX_Person_Login
on dbo.Person (UserName, EncryptedPassword)
include (/* other columns */)
create unique index IX_Person_UserName
on dbo.Person (UserName)
where (UserName is not null)
所以,现在,如果我的用户名做ID的查找,我希望的是更小,更选择性指数将优化选择。 IX_Person_UserName也应该被覆盖,因为ID是聚类关键(并且生成的计划确实承担了这一点,但这不是问题的关键)。
select ID
from dbo.Person
where UserName = @UserName
and UserName is not null
然而相反,优化选择执行索引查找上IX_Person_Login,这不是唯一的,在关键的列多,并且其叶子节点大得多。如果我强制使用IX_Person_UserName,估计的成本是相同的。在这两种情况下,估计的行数都超过了100,但实际的行数是1.我尝试更新统计数据,但是在选择的计划或估计的行数方面没有任何区别。是否因为SQL Server的计划考虑到@ UserName可能为空的可能性?即使我在查询中放置了一个字符串非空字符串值,它仍然不会使用唯一的已过滤索引。谁能解释这种行为?
尝试包括在索引中的ID – mxix 2014-09-26 16:39:02
@mxix没有必要,因为它是聚簇索引 – Lamak 2014-09-26 16:44:47
@Lamak,他想对给定的ID。由于IX_Person_Username不包含该ID,因此不会使用该ID。如果IX_Person_Username包含ID,优化器可能会为给定的查询选择它。 –
mxix
2014-09-26 16:51:48