>데이터 베이스 >MySQL 튜토리얼 >SQLServer覆盖索引

SQLServer覆盖索引

WBOY
WBOY원래의
2016-06-07 15:26:571430검색

为了更好地理解覆盖索引,在正式介绍覆盖索引之前,首先稍微来谈一谈有关索引的一些基础知识。 数据页和索引页 在SQLServer中,数据存储的基本单位是页,一页的大小为8KB,分别由页首,数据行和行偏移量组成,如下图结构: 页首固定占用96个字节,用来存储相

  为了更好地理解覆盖索引,在正式介绍覆盖索引之前,首先稍微来谈一谈有关索引的一些基础知识。

数据页和索引页

  在SQLServer中,数据存储的基本单位是页,一页的大小为8KB,分别由页首,数据行和行偏移量组成,如下图结构:

  SQLServer覆盖索引

  页首固定占用96个字节,用来存储相关的页面系统信息,例如所属的数据库表对象Id等。数据行是真实数据的存储区域,每一行的大小是不固定的。行偏移量是一个数组,数组的每个位置占2个字节,用来存储数据行距离开头的位置偏移量,主要是用来做快速定位,例如想要查找第N行,只要访问行偏移量数组的第N项,就能快速找到数据行所在的位置。索引页和数据页的结构类似,所不同的是索引页的数据行存储的是和索引相关的信息。

聚集索引和非聚集索引

  聚集索引定义了表中数据存储的真实物理位置,它是按照指定列的顺序来存储数据的,类比于新华字典中的汉字是按照拼音顺序排列的,所以每张表只能建立一个聚集索引。聚集索引是一棵B+树结构,包含索引页和数据页,最底下的一排叶子节点是数据页,往上则为索引页,来看一张图应该更清晰一些:

  SQLServer覆盖索引

  非聚集索引是独立于数据真实存储顺序逻辑而存在的,类比于新华字典中按偏旁部首查找汉字的方式。与聚集索引对比,非聚集索引也是B+树的数据结构,但却只包含索引页,而且在一张表中可以建立多个非聚集索引,有关索引的深入分析可以查看这篇文章。同样来看一张非聚集索引的图:

  SQLServer覆盖索引

什么是覆盖索引

  覆盖索引是在SQLServer2005中引入的概念,只能建立在非聚集索引的基础上,通常情况下,非聚集索引的索引页是不包含真实数据的,只存储着指向数据页中数据行的指针,而覆盖索引则是通过将数据存储在索引页上,从而在查找对应数据的时候,只要找到索引页就可以访问到数据,无需再去查询数据页,所以说这个索引是数据“覆盖”的。

SQLServer覆盖索引SQLServer覆盖索引

<span>--</span><span>覆盖索引的创建是在非聚集索引创建的基础上增加INCLUDE语句</span>
<span>CREATE</span> <span>NONCLUSTERED</span> <span>INDEX</span><span> {index_name}
</span><span>ON</span> {table_name}(column_name...) <span>--</span><span>非聚集索引可以声明指定多个列作为索引项</span>
INCLUDE(column_name...)         <span>--</span><span>覆盖索引可以指定多个列存储在索引页上</span>
创建覆盖索引的语法

覆盖索引分析

  这一小节将通过创建覆盖索引以及使用DBCC命令查看索引的方式进行介绍。

SQLServer覆盖索引SQLServer覆盖索引

<span>IF</span> <span>DB_ID</span>(<span>'</span><span>Test</span><span>'</span>) <span>IS</span> <span>NULL</span>
<span>BEGIN</span>
    <span>CREATE</span> <span>DATABASE</span><span> Test;
</span><span>END</span>
<span>GO</span>

<span>USE</span><span> Test;
</span><span>GO</span>

<span>IF</span> <span>OBJECT_ID</span>(<span>'</span><span>t1</span><span>'</span>,<span>'</span><span>U</span><span>'</span>) <span>IS</span> <span>NULL</span>
<span>BEGIN</span>
    <span>CREATE</span> <span>TABLE</span><span> t1
    (
        t1_id    </span><span>INT</span>         <span>NOT</span> <span>NULL</span> <span>IDENTITY</span>(<span>1</span>,<span>1</span><span>),
        t1_name  </span><span>VARCHAR</span>(<span>20</span>) <span>NOT</span> <span>NULL</span><span>,
        t1_name1 </span><span>VARCHAR</span>(<span>20</span>) <span>NOT</span> <span>NULL</span><span>,
        t1_name2 </span><span>VARCHAR</span>(<span>20</span>) <span>NOT</span> <span>NULL</span><span>,
        t1_name3 </span><span>VARCHAR</span>(<span>20</span>) <span>NOT</span> <span>NULL</span><span>,
        t1_name4 </span><span>VARCHAR</span>(<span>20</span>) <span>NOT</span> <span>NULL</span><span>,
        t1_name5 </span><span>VARCHAR</span>(<span>20</span>) <span>NOT</span> <span>NULL</span><span>
    );
</span><span>END</span>

<span>/*</span><span>
*插入测试数据
</span><span>*/</span>
<span>INSERT</span> <span>INTO</span><span> t1 
(   t1_name,
    t1_name1,
    t1_name2,
    t1_name3,
    t1_name4,
    t1_name5    )
    </span><span>SELECT</span> <span>'</span><span>name</span><span>'</span><span>,
           </span><span>'</span><span>name</span><span>'</span><span>,
           </span><span>'</span><span>name</span><span>'</span><span>,
           </span><span>'</span><span>name</span><span>'</span><span>,
           </span><span>'</span><span>name</span><span>'</span><span>,
           </span><span>'</span><span>name</span><span>'</span>
      <span>FROM</span><span> sysobjects o1
</span><span>CROSS</span> <span>JOIN</span><span> sysobjects o2;     

</span><span>/*</span><span>
*创建覆盖索引
</span><span>*/</span>
<span>IF</span> <span>NOT</span> <span>EXISTS</span>(<span>SELECT</span> <span>1</span>
                <span>FROM</span><span> sysindexes
               </span><span>WHERE</span> name<span>=</span><span>'</span><span>idx_t1_id</span><span>'</span><span>)
</span><span>BEGIN</span> 
    <span>CREATE</span> <span>NONCLUSTERED</span> <span>INDEX</span><span> idx_t1_id
    </span><span>ON</span><span> t1(t1_id)
    INCLUDE(t1_name);
</span><span>END</span>
创建表&测试数据&覆盖索引

  执行CROSS JOIN插入的测试数据有4000条左右,现在可以使用DBCC命令来查看表的数据页和索引页的情况。

SQLServer覆盖索引SQLServer覆盖索引

<span>/*</span><span>
*查看页的基本信息
*前提条件:表中必须插入了数据
*所需参数:(数据库名,表名,-1表示显示全部IAM页,数据页, 索引页)
</span><span>*/</span>
<span>DBCC</span> IND (Test,t1,<span>-</span><span>1</span>); 
查看表的数据页和索引页命令

  执行完这条命令后,应该可以看到显示的页信息,其中PageType=1的行表示数据页,PageType=2的行表示索引页,任意选择一条PageType=2的行,找到PageFID和PagePID,就可以使用DBCC命令来查看索引页的具体信息。

SQLServer覆盖索引SQLServer覆盖索引

<span>/*</span><span>
*查看索引页的基本信息
*所需参数:(数据库名,PageFID,PagePID,3表示输出每行每列的信息)
</span><span>*/</span>
<span>DBCC</span> PAGE(Test,<span>1</span>,<span>7732</span>,<span>3</span>);
查看索引页信息的命令

   执行完这条命令后,应该可以看到t1_name这一列的信息是包含在这个索引页中的。现在可以通过执行不同的查询SQL来查看覆盖索引所带来的性能提升,在执行SQL的同时开启显示实际的执行计划,从而可以清楚得看到对比结果。

SQLServer覆盖索引SQLServer覆盖索引

<span>SELECT</span><span> t1_name
  </span><span>FROM</span><span> t1
 </span><span>WHERE</span> t1_id <span>=</span> <span>500</span><span>;

</span><span>SELECT</span><span> t1_name1
  </span><span>FROM</span><span> t1
 </span><span>WHERE</span> t1_id <span>=</span><span>500</span>;
对比查询SQL

  SQLServer覆盖索引

  SQLServer覆盖索引

  查询1开销为33%,而查询2的开销为67%,对比可以看到查询t1_name的开销比查询t1_name1的开销小很多,因为查询t1_name只需要执行索引,就可以在索引页上找到数据,而查询t1_name1还要去查找数据页。

覆盖索引的思考

  创建索引能带来查询的优化,但却带来了更改数据的负担,覆盖索引也不意外。由上面的分析我们知道,覆盖索引是非聚集索引的进一步细化,在更新数据的时候,如果涉及到覆盖索引INCLUDE的列,除了更改数据页之外还要更改索引页,比单纯使用非聚集索引增添了额外的工作。所以,在设计覆盖索引的时候,要综合考虑应该覆盖的列,确保INCLUDE的列能带来最佳的性能优化。

성명:
본 글의 내용은 네티즌들의 자발적인 기여로 작성되었으며, 저작권은 원저작자에게 있습니다. 본 사이트는 이에 상응하는 법적 책임을 지지 않습니다. 표절이나 침해가 의심되는 콘텐츠를 발견한 경우 admin@php.cn으로 문의하세요.
이전 기사:SQLServer 触发器다음 기사:FreeRTOS 移植要点(2)