学习如何看懂SQL Server执行计划——基本知识篇

1、基本概念html

1.数据的读取数据库

  页(page)是SQL SERVER能够读写的最小I/O单位。即便只需访问一行,也要把整个页加载到缓存之中,再从缓存中读取数据。物理读取是从磁盘上读取,逻辑读取是从缓存中读取。物理读取一页的开销要比逻辑读取一页的要大得多。缓存

SET STATISTICS IO ON数据结构

--do something...oop

SET STATISTICS IO OFF性能

能够用以上代码来查看IO访问状况spa

2.表的组织方式指针

  表有两种组织方式,B树(Balance Tree)或者堆(Heap)。当在表上建立了一个汇集索引的时候,整个表数据就以B树的结构排列。不然就是按照堆的结构排列。不管表是怎么组织的,均可以在表上面建立多个非汇集索引。非汇集索引都是以B树的结构排列。htm

2.1 堆(Heap)blog

  之因此这个结构称为堆,是由于它不以任何人为指定的逻辑顺序进行排列。而是按照分区组队数据进行组织。也就是说,是按照磁盘的物理顺序。只要须要读取的数据文件没有文件系统碎片(注意和下面提到的索引的碎片区分),这个读取过程在磁盘中就能够连续的进行,没有多余的磁盘臂移动。而磁盘臂移动是I/O操做中开销最大的操做。

  堆使用一个bitmap结构来管理数据的分配。也就是它会告诉你两个结果,这个区是分配了,仍是没有分配。每个区中的物理顺序以下图。

 

  对于新插入的数据,堆只管在最后一条数据的后面的一个空闲位置保存新插入的数据,不保持任何的逻辑顺序。好比拿order表举例,若是先插入orderid 4,5,6, 假设在位置1:17六、 1:17七、1:178这三个位置。这时再插入1,这时保存的数据就变为4,5,6,1,  1保存在 1:179的位置。

 

2.2汇集索引(Clustered Index)

  该索引中键值的逻辑顺序决定了表中相应行的物理顺序。 
  汇集索引肯定表中数据的物理顺序。汇集索引相似于电话簿,后者按姓氏排列数据。因为汇集索引规定数据在表中的物理存储顺序,所以一个表只能包含一个汇集索引。但该索引能够包含多个列(组合索引),就像电话簿按姓氏和名字进行组织同样。 
     
  汇集索引对于那些常常要搜索范围值的列特别有效。使用汇集索引找到包含第一个值的行后,即可以确保包含后续索引值的行在物理相邻。例如,若是应用程序执行 的一个查询常常检索某一日期范围内的记录,则使用汇集索引能够迅速找到包含开始日期的行,而后检索表中全部相邻的行,直到到达结束日期。这样有助于提升此 类查询的性能。一样,若是对从表中检索的数据进行排序时常常要用到某一列,则能够将该表在该列上汇集(物理排序),避免每次查询该列时都进行排序,从而节 省成本。 

  当索引值惟一时,使用汇集索引查找特定的行也颇有效率。例如,使用惟一雇员 ID 列 emp_id 查找特定雇员的最快速的方法,是在 emp_id 列上建立汇集索引或 PRIMARY KEY 约束。

  继续拿Order表举例,Order表中的所有数据都保存在B树中的叶层(leaf level)中,其余层只是起到一个索引的做用,并不包含任何数据。叶层是一个双向链表结构,并按照汇集索引的主键的逻辑顺序排列。所以逻辑顺序是用指针来维护。

 

2.2.1 索引碎片

数据库中之因此会出现碎片,是由于B树的页拆分形成的。具体页拆分请参考数据结构,这里要说的是因为拆分所产生的新页不保证必定就会在被拆分的页的后面,而是可能出于文件的任何位置。这就是无序页”。换句话说,也就是在列表中处于后面位置的元素,在物理文件中却排在前面。若是你明白指针的定义的话,这句话并不难理解。由于叶层的双向列表就是以指针来维护逻辑顺序。

所以在按逻辑顺序读取的时候,因为无序页的存在,可能形成磁臂频繁的摆动。别忘记,磁盘摆动是I/O中开销最大的操做。而I/O每每是一个系统的瓶颈所在。

若是按照物理顺序来读取,也就是unordered读取,就会避免上面所产生的问题。再次强调,unordered是指不按逻辑顺序读取,因此叫unordered。

2.2.2 索引的层数

索引的层数,也就是B树的高度,直接代表了一次查找操做在页面读取方面的开销。一些执行计划如Nested loop联接会屡次调用查找操做。所以理解这个概念很重要。

树的高度主要和如下几个因素相关

1.表的总行数。

2.平均一行保存数据的大小。

3.页的平均密度。由于不是每一页都应该填充满数据,这样能够减小页拆分的次数。

4.一页所能容纳的行数。

具体公式也很简单,3级索引大概能容纳4百万行,4级索引大概能容纳4亿行数据。所以一般一张表的索引层数一般为3到4级。 

 

2.3非汇集索引(NonClustered Index)

  该索引中索引的逻辑顺序与磁盘上行的物理存储顺序不一样。 

  索引是经过二叉树的数据结构来描述的,咱们能够这么理解聚簇索引:索引的叶节点就是数据节点。而非聚簇索引的叶节点仍然是索引节点,只不过有一个指针指向对应的数据块。

  和汇集索引的区别就在于它的叶层并不包含全部的数据。在默认状况下它只包含了键列的数据,并包含了一个行定位符(row locator)。这个行定位符的具体内容取决于它创建在以堆形式的表仍是以B树组织的表,换句话说也就是这张表是否创建了汇集索引会影响到非汇集索引的行定位符。若是是创建了汇集索引,那么这个行定位符就是一个汇集键,咱们经过这个汇集键再次查找汇集索引上的数据。

 

2.3.1 若是非汇集索引包含了咱们须要查找的全部数据

  这种状况咱们一般叫作索引覆盖。

  正由于非汇集索引有着和索引同样的结构,而且因为非汇集索引所包含的列少,所以数据量就小,使得叶层的一页能包含更多的行,所以进行一次I/O页读取的动做的时候,就能读取进更多的行。所以查找效率是最高的。

  举个不恰当的例子,美女征婚,应征人员的我的信息表有 “姓名、 德、 智、 体 、美、 劳、 高、 富、 帅”这几列,按姓名排序。美女只关注“高、 富、 帅”这三列的内容,为了更快的筛选,咱们帮美女按照我的信息表的内容从新制做了一张表,这张表忽略了其余信息,只保留了高、富、帅和姓名,筛选效率固然就比原来关注更多内容时要高。

 

2.3.2 若是非汇集索引不包含咱们须要查找的全部数据

  通俗的说这时咱们就须要从非汇集索引中所包含的线索去包含全部数据的表中去找。

  按照咱们以前的定义换句话来讲,就是经过非汇集索引中的行定位符去汇集索引或者堆中去查找所需的数据。

 

2、深刻浅出理解索引结构

  实际上,您能够把索引理解为一种特殊的目录。微软的SQL SERVER提供了两种索引:汇集索引(clustered index,也称聚类索引、簇集索引)和非汇集索引(nonclustered index,也称非聚类索引、非簇集索引)。下面,咱们举例来讲明一下汇集索引和非汇集索引的区别:
  其实,咱们的汉语字典的正文自己就是一个汇集索引。好比,咱们要查“安”字,就会很天然地翻开字典的前几页,由于“安”的拼音是“an”,而按照拼音排序汉字的字典是以英文字母“a”开头并以“z”结尾的,那么“安”字就天然地排在字典的前部。若是您翻完了全部以“a”开头的部分仍然找不到这个字,那么就说明您的字典中没有这个字;一样的,若是查“张”字,那您也会将您的字典翻到最后部分,由于“张”的拼音是“zhang”。也就是说,字典的正文部分自己就是一个目录,您不须要再去查其余目录来找到您须要找的内容。咱们把这种正文内容自己就是一种按照必定规则排列的目录称为“汇集索引”。
  若是您认识某个字,您能够快速地从自动中查到这个字。但您也可能会遇到您不认识的字,不知道它的发音,这时候,您就不能按照刚才的方法找到您要查的字,而须要去根据“偏旁部首”查到您要找的字,而后根据这个字后的页码直接翻到某页来找到您要找的字。但您结合“部首目录”和“检字表”而查到的字的排序并非真正的正文的排序方法,好比您查“张”字,咱们能够看到在查部首以后的检字表中“张”的页码是672页,检字表中“张”的上面是“驰”字,但页码倒是63页,“张”的下面是“弩”字,页面是390页。很显然,这些字并非真正的分别位于“张”字的上下方,如今您看到的连续的“驰、张、弩”三字实际上就是他们在非汇集索引中的排序,是字典正文中的字在非汇集索引中的映射。咱们能够经过这种方式来找到您所须要的字,但它须要两个过程,先找到目录中的结果,而后再翻到您所须要的页码。咱们把这种目录纯粹是目录,正文纯粹是正文的排序方式称为“非汇集索引”。
  经过以上例子,咱们能够理解到什么是“汇集索引”和“非汇集索引”。进一步引伸一下,咱们能够很容易的理解:每一个表只能有一个汇集索引,由于目录只能按照一种方法进行排序。

一、什么时候使用汇集索引或非汇集索引

下面的表总结了什么时候使用汇集索引或非汇集索引(很重要):

 

动做描述

使用汇集索引

使用非汇集索引

列常常被分组排序

返回某范围内的数据

不该

一个或极少不一样值

不该

不该

小数目的不一样值

不该

大数目的不一样值

不该

频繁更新的列

不该

外键列

主键列

频繁修改索引列

不该



      事实上,咱们能够经过前面汇集索引和非汇集索引的定义的例子来理解上表。如:返回某范围内的数据一项。好比您的某个表有一个时间列,刚好您把聚合索引创建在了该列,这时您查询2004年1月1日至2004年10月1日之间的所有数据时,这个速度就将是很快的,由于您的这本字典正文是按日期进行排序的,聚类索引只须要找到要检索的全部数据中的开头和结尾数据便可;而不像非汇集索引,必须先查到目录中查到每一项数据对应的页码,而后再根据页码查到具体内容。

二、结合实际,谈索引使用的误区

      理论的目的是应用。虽然咱们刚才列出了什么时候应使用汇集索引或非汇集索引,但在实践中以上规则却很容易被忽视或不能根据实际状况进行综合分析。下面咱们将根据在实践中遇到的实际问题来谈一下索引使用的误区,以便于你们掌握索引创建的方法。

     2.一、主键就是汇集索引
      这种想法笔者认为是极端错误的,是对汇集索引的一种浪费。虽然SQL SERVER默认是在主键上创建汇集索引的。
      一般,咱们会在每一个表中都创建一个ID列,以区分每条数据,而且这个ID列是自动增大的,步长通常为1。咱们的这个办公自动化的实例中的列Gid就是如此。此时,若是咱们将这个列设为主键,SQL SERVER会将此列默认为汇集索引。这样作有好处,就是可让您的数据在数据库中按照ID进行物理排序,但笔者认为这样作意义不大。
      显而易见,汇集索引的优点是很明显的,而每一个表中只能有一个汇集索引的规则,这使得汇集索引变得更加珍贵。
      从咱们前面谈到的汇集索引的定义咱们能够看出,使用汇集索引的最大好处就是可以根据查询要求,迅速缩小查询范围,避免全表扫描。在实际应用中,由于 ID号是自动生成的,咱们并不知道每条记录的ID号,因此咱们很难在实践中用ID号来进行查询。这就使让ID号这个主键做为汇集索引成为一种资源浪费。其次,让每一个ID号都不一样的字段做为汇集索引也不符合“大数目的不一样值状况下不该创建聚合索引”规则;固然,这种状况只是针对用户常常修改记录内容,特别是索引项的时候会负做用,但对于查询速度并无影响。
      在办公自动化系统中,不管是系统首页显示的须要用户签收的文件、会议仍是用户进行文件查询等任何状况下进行数据查询都离不开字段的是“日期”还有用户自己的“用户名”。
      一般,办公自动化的首页会显示每一个用户还没有签收的文件或会议。虽然咱们的where语句能够仅仅限制当前用户还没有签收的状况,但若是您的系统已创建了很长时间,而且数据量很大,那么,每次每一个用户打开首页的时候都进行一次全表扫描,这样作意义是不大的,绝大多数的用户1个月前的文件都已经浏览过了,这样作只能徒增数据库的开销而已。事实上,咱们彻底可让用户打开系统首页时,数据库仅仅查询这个用户近3个月来未阅览的文件,经过“日期”这个字段来限制表扫描,提升查询速度。若是您的办公自动化系统已经创建的2年,那么您的首页显示速度理论上将是原来速度8倍,甚至更快。
      在这里之因此提到“理论上”三字,是由于若是您的汇集索引仍是盲目地建在ID这个主键上时,您的查询速度是没有这么高的,即便您在“日期”这个字段上创建的索引(非聚合索引)。下面咱们就来看一下在1000万条数据量的状况下各类查询的速度表现(3个月内的数据为25万条):

    (1)仅在主键上创建汇集索引,而且不划分时间段:

    Select gid,fariqi,neibuyonghu,title from tgongwen

    用时:128470毫秒(即:128秒)

    (2)在主键上创建汇集索引,在fariq上创建非汇集索引:

    select gid,fariqi,neibuyonghu,title from Tgongwen
    where fariqi> dateadd(day,-90,getdate())

    用时:53763毫秒(54秒)

    (3)将聚合索引创建在日期列(fariqi)上:

    select gid,fariqi,neibuyonghu,title from Tgongwen
    where fariqi> dateadd(day,-90,getdate())

    用时:2423毫秒(2秒)

      虽然每条语句提取出来的都是25万条数据,各类状况的差别倒是巨大的,特别是将汇集索引创建在日期列时的差别。事实上,若是您的数据库真的有1000 万容量的话,把主键创建在ID列上,就像以上的第一、2种状况,在网页上的表现就是超时,根本就没法显示。这也是我摒弃ID列做为汇集索引的一个最重要的因素。得出以上速度的方法是:在各个select语句前加:

    declare @d datetime
    set @d=getdate()

    并在select语句后加:

    select [语句执行花费时间(毫秒)]=datediff(ms,@d,getdate())

    2.二、只要创建索引就能显著提升查询速度
      事实上,咱们能够发现上面的例子中,第二、3条语句彻底相同,且创建索引的字段也相同;不一样的仅是前者在fariqi字段上创建的是非聚合索引,后者在此字段上创建的是聚合索引,但查询速度却有着天壤之别。因此,并不是是在任何字段上简单地创建索引就能提升查询速度。
      从建表的语句中,咱们能够看到这个有着1000万数据的表中fariqi字段有5003个不一样记录。在此字段上创建聚合索引是再合适不过了。在现实中,咱们天天都会发几个文件,这几个文件的发文日期就相同,这彻底符合创建汇集索引要求的:“既不能绝大多数都相同,又不能只有极少数相同”的规则。由此看来,咱们创建“适当”的聚合索引对于咱们提升查询速度是很是重要的。

    2.三、把全部须要提升查询速度的字段都加进汇集索引,以提升查询速度
      上面已经谈到:在进行数据查询时都离不开字段的是“日期”还有用户自己的“用户名”。既然这两个字段都是如此的重要,咱们能够把他们合并起来,创建一个复合索引(compound index)。
      不少人认为只要把任何字段加进汇集索引,就能提升查询速度,也有人感到迷惑:若是把复合的汇集索引字段分开查询,那么查询速度会减慢吗?带着这个问题,咱们来看一下如下的查询速度(结果集都是25万条数据):(日期列fariqi首先排在复合汇集索引的起始列,用户名neibuyonghu排在后列):

    (1)select gid,fariqi,neibuyonghu,title from Tgongwen where fariqi>''2004-5-5''

    查询速度:2513毫秒

    (2)select gid,fariqi,neibuyonghu,title from Tgongwen
                where fariqi>''2004-5-5'' and neibuyonghu=''办公室''

    查询速度:2516毫秒

    (3)select gid,fariqi,neibuyonghu,title from Tgongwen where neibuyonghu=''办公室''

    查询速度:60280毫秒

      从以上试验中,咱们能够看到若是仅用汇集索引的起始列做为查询条件和同时用到复合汇集索引的所有列的查询速度是几乎同样的,甚至比用上所有的复合索引列还要略快(在查询结果集数目同样的状况下);而若是仅用复合汇集索引的非起始列做为查询条件的话,这个索引是不起任何做用的。固然,语句一、2的查询速度同样是由于查询的条目数同样,若是复合索引的全部列都用上,并且查询结果少的话,这样就会造成“索引覆盖”,于是性能能够达到最优。同时,请记住:不管您是否常用聚合索引的其余列,但其前导列必定要是使用最频繁的列。

三、其余书上没有的索引使用经验总结

     3.一、用聚合索引比用不是聚合索引的主键速度快
      下面是实例语句:(都是提取25万条数据)

    select gid,fariqi,neibuyonghu,reader,title from Tgongwen where fariqi=''2004-9-16''

    使用时间:3326毫秒

    select gid,fariqi,neibuyonghu,reader,title from Tgongwen where gid<=250000

    使用时间:4470毫秒

    这里,用聚合索引比用不是聚合索引的主键速度快了近1/4。

     3.二、用聚合索引比用通常的主键做order by时速度快,特别是在小数据量状况下

    select gid,fariqi,neibuyonghu,reader,title from Tgongwen order by fariqi

    用时:12936

    select gid,fariqi,neibuyonghu,reader,title from Tgongwen order by gid

    用时:18843

      这里,用聚合索引比用通常的主键做order by时,速度快了3/10。事实上,若是数据量很小的话,用汇集索引做为排序列要比使用非汇集索引速度快得明显的多;而数据量若是很大的话,如10万以上,则两者的速度差异不明显。

     3.三、使用聚合索引内的时间段,搜索时间会按数据占整个数据表的百分比成比例减小,而不管聚合索引使用了多少个:

    select gid,fariqi,neibuyonghu,reader,title from Tgongwen where fariqi>''2004-1-1''

    用时:6343毫秒(提取100万条)

    select gid,fariqi,neibuyonghu,reader,title from Tgongwen where fariqi>''2004-6-6''

    用时:3170毫秒(提取50万条)

    select gid,fariqi,neibuyonghu,reader,title from Tgongwen where fariqi=''2004-9-16''

    用时:3326毫秒(和上句的结果如出一辙。若是采集的数量同样,那么用大于号和等于号是同样的)

    select gid,fariqi,neibuyonghu,reader,title from Tgongwen
                where fariqi>''2004-1-1'' and fariqi<''2004-6-6''

    用时:3280毫秒

     3.四、日期列不会由于有分秒的输入而减慢查询速度
      下面的例子中,共有100万条数据,2004年1月1日之后的数据有50万条,但只有两个不一样的日期,日期精确到日;以前有数据50万条,有5000个不一样的日期,日期精确到秒。

    select gid,fariqi,neibuyonghu,reader,title from Tgongwen
              where fariqi>''2004-1-1'' order by fariqi

    用时:6390毫秒

    select gid,fariqi,neibuyonghu,reader,title from Tgongwen
                where fariqi<''2004-1-1'' order by fariqi

    用时:6453毫秒

四、其余注意事项

      “水可载舟,亦可覆舟”,索引也同样。索引有助于提升检索性能,但过多或不当的索引也会致使系统低效。由于用户在表中每加进一个索引,数据库就要作更多的工做。过多的索引甚至会致使索引碎片。
      因此说,咱们要创建一个“适当”的索引体系,特别是对聚合索引的建立,更应精益求精,以使您的数据库能获得高性能的发挥。

 

总结转载与如下博客中

http://www.cnblogs.com/lwzz/archive/2012/08/05/2620824.html 

http://www.cnblogs.com/aspnethot/articles/1504082.html

转载自:http://www.cnblogs.com/taiyonghai/p/5780907.html

相关文章
相关标签/搜索