3 回答
TA贡献1943条经验 获得超7个赞
看一下这样的索引:
Cols
1 2 3
-------------
| | 1 | |
| A |---| |
| | 2 | |
|---|---| |
| | | |
| | 1 | 9 |
| B | | |
| |---| |
| | 2 | |
| |---| |
| | 3 | |
|---|---| |
看到第一列限制比第一列限制消除更多结果的方式如何限制第一列?如果您想象必须如何遍历索引,第1列然后第2列,等等,会更容易。。。您会发现,第一遍中大部分结果的丢失使第二步快得多。
另一种情况,如果您查询第3列,则优化器甚至都不会使用索引,因为它对缩小结果集完全没有帮助。 随时在查询中,在进行下一步之前,缩小要处理的结果数的范围即可提高性能。
由于索引也是以这种方式存储的,因此在查询索引时,不会在索引上回溯以找到第一列。
简而言之:不,这不是为了展示,而是有真正的性能优势。
TA贡献1801条经验 获得超16个赞
列的顺序至关重要。现在哪个顺序正确,取决于您要查询的顺序。索引可用于执行精确搜索或范围扫描。精确查找是指指定索引中所有列的值,并且查询恰好位于该行所关注的位置。对于查找,列的顺序无关紧要。范围扫描是仅指定一些列的情况,在这种情况下,顺序变得很重要。只有指定了最左列,然后指定了下一个最左列,SQL Server才可以将索引用于范围扫描。如果您对(A,B,C)的索引可以用来范围扫描A=@a
,为A=@a AND B=@b
但不为B=@b
,对于C=@c
也不B=@b AND C=@c
。情况A=@a AND C=@c
是混合的,例如A=@a
部分将使用索引,但C=@c
不使用索引(查询将扫描所有B值的A=@a
,不会“跳到” C=@c
)。其他数据库系统具有所谓的“跳过扫描”运算符,当未指定外部列时可以利用索引中的内部列。
掌握了这些知识之后,您可以再次查看索引定义。(MostSelective, SecondMost, Least)
仅当MostSelective
指定column 时,索引on 才有效。但是,最有选择性的是,内部列的相关性将迅速降低。很多时候,您会发现(MostSelective) include (SecondMost, Least)
或上有一个更好的索引(MostSelective, SecondMost) include (Least)
。由于内部列的相关性较低,因此将低选择性列放置在索引中的此类正确位置上只会使它们产生寻找噪音,因此将它们移出中间页并仅将它们保留在叶子页上是有意义的,因为查询覆盖率的目的。换句话说,将它们移动到INCLUDE。随着Least
列大小的增加,这一点变得更加重要。想法是该索引只能使指定以下内容的查询受益MostSelective
无论是精确值还是范围,该列的选择性最高,已经在很大程度上限制了候选行。
另一方面,上的索引(Least, SecondMost, MostSelective)
看似错误,但实际上它是一个功能强大的索引。因为它具有Least
列作为其最外面的查询,所以它可用于必须在低选择性列上聚合结果的查询。这样的查询在OLAP和分析数据仓库中很普遍,而这正是此类索引非常适合的地方。这样的索引实际上是出色的聚集索引,正是因为它们将物理布局组织在大块相关行上(相同的Least
值,通常表示某种类别或类型),并且有助于分析查询。
因此,不幸的是,没有“正确”的命令。您不应该遵循任何曲奇工具的配方,而应针对这些表分析将要使用的查询模式,并确定哪个索引列顺序正确。
添加回答
举报