我们在学sqlserver的时候,大多教科书和前辈们都说状态少的字段不要建索引,由此带来的开销还不如不建索引,但是这句话有多少人真的知道,或者说有多少人真的对此有比较深刻的理解,而不是听别人道听途说。。。这样记得快,忘记的也不慢。。。这篇我来分析一下这句话到底有几个意思。

一:现象

首先我们还是用测试数据来发现问题,我先建立一个Person,有5个字段,建表sql如下:


DROP TABLE dbo.PersonCREATE TABLE Person(ID INT PRIMARY KEY IDENTITY,NAME VARCHAR(900),Age INT,Email VARCHAR(20),isMan INT )-- 在isMan字段创建非聚集索引(0:女 1:男)
CREATE INDEX idx_isMan ON dbo.Person(isMan)DECLARE @ch AS INT=0WHILE @ch<=100000
BEGININSERT INTO dbo.Person(NAME,Age,Email,isMan)VALUES(REPLICATE(CHAR(@ch),50),@ch,CAST(CAST(RAND()*1000000000 AS INT) AS VARCHAR(10))+'qq.com',@ch%2)SET @ch=@ch+1
END

通过上面的sql发现ID为聚集索引,isMan为非聚集索引,isMan也就是两种状态(0,1),并且插入10w条记录,截图如下:

sql都做完了,接下来要做的事情就是查询下:isMan=1的记录,如下图:

我靠。。。我明明是在isMan上做数据检索的,怎么就变成 “聚集索引扫描”了???这sqlserver什么意思嘛,居然不走我的“idx_isMan”索引,却走他的“聚集索引(PKPerson3214EC276EF57B66)”。。。。同时也看到上面的”逻辑读取”为521。。。说明在内存中走了521个数据页。但是我不服呀。。。我一定要让执行计划走我的索引。。。办法就是强制指定。。。如下图。

看到上面的图,你是不是已经疯了。。。我才捞5w的数据,你给我走了10w多次数据页。。。这么说1条记录要走两个数据页。。。而扫描聚集索引才走521个数据页,相差200倍。。。难怪执行计划打死也不走“idx_isMan”这条索引。。。

二:分析原因

现在很生气,整个人都不好了,为什么会这样???为了找出问题,我们还得看数据页。


DBCC TRACEON(3604,2588)
DBCC IND(Ctrip,Person,-1)

通过上面的三个图,大概可以看到,10w条数据用了697数据页,其中聚集索引有521个,非聚集索引为176个,这也说明了上面的”聚集索引扫描“走遍了它自己所有的数据页来才捞出数据,同时还发现这两个索引都有一个共同特征就是,只有一个根节点(indexLevel=1)和无数个(indexLevel=0)叶子节点,然后我脑子里面就有一幅图出来了。。。

上面就是我构思出来的图,这个专业一点的名字叫做书签查找。。。我们通过建立”idx_isMan“索引后,就会构建右半图的B树结构,其中索引记录会存放两个值,一个是索引值isMan和一个聚集索引值ID,如果你不相信的话,可以通过DBCC Page去探索"idx_isMan"的索引页,你也可以通过DBCC SHOW_STATISTICS 去查看,如图:

然后引擎通过“idx_isMan“扫描后,拿到了key值,但是非常可惜,我是select * 的,所以必须还要喷出记录中的Name,Emai等l字段,但是”index_isMan"中并没有保存这几个字段,所以必须通过key去”聚集索引“的B树中去找。。。最后通过”聚集索引“的B树找到了目标记录,这也就是所谓的执行计划中的”键查找“,然后喷出”Name,Email“等字段。。。。问题就在这里。。。因为我这样来回的蹦跶蹦跶。。。造成了找出完整的一个记录,需要蹦跶2-3次数据页。。。具体的寻找记录,可参考图中的”紫色线条“,最后也就造成了10w多次蹦跶。。。

三:启示

那这个例子给我们什么启示呢???仔细想想你就知道。。。使用非聚集索引,千万不要捞取过多的数据。。。因为过多的数据会造成在多个B树中来回的蹦跶。。。想要做到捞取数据较少,就必须在高唯一性的字段上建立索引,这样的话在非聚集索引B树中符合的数据相对较少,也就减少了我蹦跶到”主键索引“的B树次数。。。这样的话来回蹦跶的次数远远比”聚集索引“扫描来的实惠,对不对。。。

四:结论

必须在唯一性较高的字段上建立非聚集索引。

Sql Server之旅——第六站 为什么都说状态少的字段不能建索引相关推荐

  1. sqlserver怎么查看索引_Sql Server之旅——第六站 为什么都说状态少的字段不能建索引...

    我们在学sqlserver的时候,大多教科书和前辈们都说状态少的字段不要建索引,由此带来的开销还不如不建索引,但是这句话有多少人真的知道,或者说有多少人真的对此有比较深刻的理解,而不是听别人道听途说. ...

  2. Sql Server之旅——第八站 看公司这些DBA们设计的这些复合索引

    这一篇再说下索引的最后一个主题,索引覆盖,当然学习比较好的捷径是看看那些大师们设计的索引,看从中能提取些什么营养的东西,下面我们看看数据库中一个核心的Orders表. 一:查看表的架构 1. 先查看这 ...

  3. Sql Server之旅——第十一站 简单说说sqlserver的执行计划

    原文:Sql Server之旅--第十一站 简单说说sqlserver的执行计划 我们知道sql在底层的执行给我们上层人员开了一个窗口,那就是执行计划,有了执行计划之后,我们就清楚了那些烂sql是怎么 ...

  4. Sql Server之旅——第五站 确实不得不说的DBCC命令

    Sql Server之旅--第五站 确实不得不说的DBCC命令 原文:Sql Server之旅--第五站 确实不得不说的DBCC命令 今天研发中心办年会,晚上就是各自部门聚餐了,我个人喜欢喝干红,在干 ...

  5. Sql Server之旅——第四站 你必须知道的非聚集索引扫描

    非聚集索引,这个是大家都非常熟悉的一个东西,有时候我们由于业务原因,sql写的非常复杂,需要join很多张表,然后就泪流满面了...这时候就有DBA或者资深的开发给你看这个猥琐的sql,通过执行计划一 ...

  6. Sql Server之旅——第十站 简单说说sqlserver的执行计划

    我们知道sql在底层的执行给我们上层人员开了一个窗口,那就是执行计划,有了执行计划之后,我们就清楚了那些烂sql是怎么执行的,这样 就可以方便的找到sql的缺陷和优化点. 一:执行计划生成过程 说到执 ...

  7. Sql Server之旅——第七站 复合索引和include索引到底有多大区别?

    索引和锁,这两个主题对我们开发工程师来说,非常的重要...只有理解了这两个主题,我们才能写出高质量的sql语句,在之前的博客中,我所说的索引都是单列索引...当然数据库不可能只认单列索引,还有我这篇的 ...

  8. Sql Server之旅——第十三站 深入的探讨锁机制

    上一篇我只是做了一个堆表让大家初步的认识到锁的痉挛状态,但是在现实世界上并没有这么简单的事情,起码我的表不会没有索引对吧,还有就是我的表一定会有很多的连接过来,10:1的读写,很多码农可能都会遇到类似 ...

  9. Sql Server之旅——第三站 解惑那些背了多年聚集索引的人

    说到聚集索引,我想每个码农都明白,但是也有很多像我这样的伪程序员,只能用死记硬背来解决这个问题,什么表中只能建一个聚集索引,然后又扯到了目录查找来帮助读者记忆....问题就在这里,我们不是学文科,,, ...

最新文章

  1. 和12岁小同志搞创客开发:Mind+编程软件简介、安装及使用
  2. python要学哪些_学python都要学哪些内容?
  3. Spring MVC 特性实现文件下载
  4. 1049.(*) Counting Ones
  5. 数字经济的核心是对大数据_大数据崛起为数字世界的核心润滑剂
  6. echart 动画 饼图_echarts构建关系图,节点可收缩和展开,可添加点击事件
  7. 【学堂在线数据挖掘:理论方法笔记】第九天(4.3)
  8. 訪问可能没有定义的data (通过static类型flash.net:FileReference引用)
  9. Android的Matrix 2
  10. 小白学VUE——实现抖音时钟(CDN方式)
  11. ES6阮一峰读书笔记第二章变量的解构赋值
  12. 服务器游戏性能测试工具,python 游戏服务器 性能测试工具
  13. Service Mesh对比:Istio与Linkerd
  14. . 显示隐藏文件夹选项消失时如何还原
  15. 什么是CISSP(国际注册信息安全认证专家)?
  16. 成都中忻嘉业:抖音小店商品不出单,如何做好优化
  17. 算法017:三步问题。有个小孩正在上楼梯,楼梯有n阶台阶,小孩一次可以上1阶、2阶或3阶。实现一种方法,计算小孩有多少种上楼梯的方式。结果可能很大,你需要对结果模1000000007。
  18. 三国志战略版:开荒实录系列—关妹张飞,S11官渡之战开荒
  19. MLOps极致细节:17. Azure ML Pipeline(机器学习管道),模型训练,打包和注册
  20. Onvif协议客户端开发(1)--Onvif协议介绍

热门文章

  1. 闭包 —JavaScript面向对象高级
  2. [No000022]他们说:得诺贝尔奖到底有多难?
  3. BZOJ 2588: Spoj 10628. Count on a tree 树上跑主席树
  4. 基于visual Studio2013解决C语言竞赛题之0710排序函数
  5. JavaScript自动设置IFrame高度(兼容各主流浏览器)
  6. RFID会议签到系统总结(二十一)――服务端的通讯
  7. 洛谷——P1305 新二叉树(新建二叉树以及遍历)
  8. 十年老站吐血迁移实录
  9. OpenGL® ES 3.0 Programming Guide - Book Website
  10. 基于Azure Blob冷存储的数据压缩备份总结