SQL Server数据库如何执行计划及Sql查询优化
沉沙 2018-05-09 来源 : 阅读 2038 评论 0

摘要:本文主要讲解SQL Server数据库如何执行计划及Sql查询优化,帮助大家知道如何查看SQL的优化机制,以此来优化SQL查询,而不是仅仅用程序执行结果来优化。

今天来讨论下SQL的执行计划,来让大家知道如何查看SQL的优化机制,以此来优化SQL查询,而不是仅仅用程序执行结果来优化。

--DROP TABLE T_UserInfo--------------------------------------建测试表 
CREATE TABLE T_UserInfo 
( 
    Userid varchar(20),  UserName varchar(20), 
    RegTime datetime, Tel varchar(20), 
) 
--插入测试数据 
DECLARE @I INT 
DECLARE @ENDID INT 
SELECT @I = 1 
SELECT @ENDID = 100  --在此处更改要插入的数据,重新插入之前要删掉所有数据 
WHILE @I <= @ENDID 
BEGIN 
    INSERT INTO T_UserInfo  
    SELECT ’ABCDE’+CAST(@I AS VARCHAR(20))+’EF’,’李’+CAST(@I AS VARCHAR(20)),GETDATE(),’876543’+CAST(@I AS VARCHAR(20)) 
    SELECT @I = @I + 1 
END



--相关SQL语句解释 
-------------------------------------------------------------建聚集索引 
CREATE CLUSTERED INDEX INDEX_Userid  ON T_UserInfo (Userid) 
--建非聚集索引 
CREATE NONCLUSTERED INDEX INDEX_Userid  ON T_UserInfo (Userid) 
--删除索引 
DROP INDEX T_UserInfo.INDEX_Userid 
----------------------------------------------------------- 
-------------------------------------------------------------
显示有关由Transact-SQL 语句生成的磁盘活动量的信息 
SET STATISTICS IO ON 
--关闭有关由Transact-SQL 语句生成的磁盘活动量的信息 
SET STATISTICS IO OFF 
--显示[返回有关语句执行情况的详细信息,并估计语句对资源的需求] 
SET SHOWPLAN_ALL  ON  
--关闭[返回有关语句执行情况的详细信息,并估计语句对资源的需求] 
SET SHOWPLAN_ALL  OFF 
----------------------------------------------------------- 
请记住:SET STATISTICS IO 和 SET SHOWPLAN_ALL 是互斥的。

OK,现在开始: 首先,我们插入100条数据。然后写一个查询语句: 

SELECT * FROM T_UserInfo WHERE USERID='ABCDE6EF'

然后我们来看该语句对IO的读写: 
执行:

SET STATISTICS IO ON

此时再执行该SQL:

SELECT * FROM T_UserInfo WHERE USERID='ABCDE6EF'

切换到消息栏显示如下: 
    表'T_UserInfo'。扫描计数1,逻辑读1 次,物理读0 次,预读0 次。 
解释下其意思: 
    四个值分别为:     
        执行的扫描次数;     
        从数据缓存读取的页数;     
        从磁盘读取的页数;     
        为进行查询而放入缓存的页数 
重要:如果对于一个SQL查询有多种写法,那么这四个值中的逻辑读(logical reads)决定了哪个是最优化的。   

接下来我们为其建一个聚集索引 
执行

CREATE CLUSTERED INDEX INDEX_Userid  ON T_UserInfo (Userid)

然后再执行

SELECT * FROM T_UserInfo WHERE USERID='ABCDE6EF'

切换到消息栏如下显示: 
    表'T_UserInfo'。扫描计数1,逻辑读2 次,物理读0 次,预读0 次。 
此时逻辑读由原来的1变成2, 说明我们又加了一个索引页,现在我们查询时,逻辑读就是要读两页(1索引页+1数据页),此时的效率还不如不建索引。   
此时再选中查询语句,然后再Ctrl+L,如下图:  
聚集索引查找:扫描聚集索引中特定范围的行 说明,此时用了索引。   
OK,到这里你应该已经知道初步知道MSSQL查询计划和如何查看对IO的读取消耗了吧!     

接下来我们继续:   
现在我再把测试数据改变成1000条 
再执行

SET STATISTICS IO ON,

再执行

SELECT * FROM T_UserInfo WHERE USERID='ABCDE6EF'

在不加聚集索引的情况下: 
    表'T_UserInfo'。扫描计数1,逻辑读7 次,物理读0 次,预读0 次。 
    在加聚集索引的情况下:

CREATE CLUSTERED INDEX INDEX_Userid  ON T_UserInfo (Userid)

    表'T_UserInfo'。扫描计数1,逻辑读2 次,物理读0 次,预读0 次。 
(其实也就是说此时是读了一个索引页,一个数据页) 如此,在数据量稍大时,索引的查询优势就显示出来了。       

先小总结下: 当你构建SQL语句时,按Ctrl+L就可以看到语句是如何执行,是用索引扫描还是表扫描? 通过SET STATISTICS IO ON 来查看逻辑读,完成同一功能的不同SQL语句,逻辑读 越小查询速度越快(当然不要找那个只有几百条记录的例子来反我)。    

我们再继续深入: 
OK,现在我们再来看一次,我们换个SQL语句,来看下MSSQL如何来执行的此SQL呢? 
现在去掉索引:DROP INDEX T_UserInfo.INDEX_Userid 
现在打开[显示语句执行情况的详细信息]:SET SHOWPLAN_ALL  ON 
然后再执行:SELECT * FROM T_UserInfo WHERE USERID LIKE 'ABCDE8%' 
看结果栏:结果中有些具体参数,比如IO的消耗,CPU的消耗。 
在这里我们只看StmtText: 

SELECT * FROM T_UserInfo WHERE USERID LIKE 'ABCDE8%'   
    |--Table Scan(OBJECT:([student].[dbo].[T_UserInfo]), WHERE:(like([T_UserInfo].[Userid], 'ABCDE8%', NULL)))


再加上索引: 
先关闭:

SET SHOWPLAN_ALL OFF

再执行:

CREATE CLUSTERED INDEX INDEX_Userid  ON T_UserInfo (Userid)

再开启:

SET SHOWPLAN_ALL ON

再执行:

SELECT * FROM T_UserInfo WHERE USERID LIKE 'ABCDE8%'

查看StmtText: 

SELECT * FROM T_UserInfo WHERE USERID LIKE 'ABCDE8%'   
    |--Clustered Index Seek(OBJECT:([student].[dbo].[T_UserInfo].[INDEX_Userid]), SEEK:([T_UserInfo].[Userid] >= 'ABCDE8' AND [T_UserInfo].[Userid] < 'ABCDE9'),  WHERE:(like([T_UserInfo].[Userid], 'ABCDE8%', NULL)) ORDERED FORWARD)

在有索引的情况下,我们再写一个SQL: 

SET SHOWPLAN_ALL ON SELECT * FROM T_UserInfo WHERE LEFT(USERID,4)='ABCDE8%'

查看StmtText: 

SELECT * FROM T_UserInfo WHERE LEFT(USERID,4)='ABCDE8%'   
    |--Clustered Index Scan(OBJECT:([student].[dbo].[T_UserInfo].[INDEX_Userid]), WHERE:(substring([T_UserInfo].[Userid], 1, 4)='ABCDE8%'))

我们再分别看一下三种情况下对IO的操作 分别如下: 
第一种情况:表'T_UserInfo'。扫描计数1,逻辑读7 次,

物理读0 次,预读0 次。 
第二种情况:表'T_UserInfo'。扫描计数1,逻辑读3 次,物理读0 次,预读0 次。 
第三种情况:表'T_UserInfo'。扫描计数1,逻辑读8 次,物理读0 次,预读0 次。 

这说明: 
第一次是表扫描,扫了7页,也就是全表扫描 
第二次是索引扫描,扫了1页索引,2页数据页 
第三次是索引扫描+表扫描,扫了1页索引,7页数据页 
[图形界面也有对CPU和IO的消耗,也可以看出来哪个最优!]    

通过比较,很容易的看出:第二种第三种写法在都有索引的情况下,like有效的使用索引,而left则不能,这样一个最简单的优化的例子就出来了。  

  
如果以上你都明白了,那么你可能已经对SQL的优化有初步新的想法了,网上一堆堆的SQL优化的文章真的是那样吗?你自己试试就知道了,而不必盲目去记那些东西,自己试试,了解更多详情请关注数据库SQL Server频道!


本文由 @沉沙 发布于职坐标。未经许可,禁止转载。
喜欢 | 0 不喜欢 | 0
看完这篇文章有何感觉?已经有0人表态,0%的人喜欢 快给朋友分享吧~
评论(0)
后参与评论

您输入的评论内容中包含违禁敏感词

我知道了

助您圆梦职场 匹配合适岗位
验证码手机号,获得海同独家IT培训资料
选择就业方向:
人工智能物联网
大数据开发/分析
人工智能Python
Java全栈开发
WEB前端+H5

请输入正确的手机号码

请输入正确的验证码

获取验证码

您今天的短信下发次数太多了,明天再试试吧!

提交

我们会在第一时间安排职业规划师联系您!

您也可以联系我们的职业规划师咨询:

小职老师的微信号:z_zhizuobiao
小职老师的微信号:z_zhizuobiao

版权所有 职坐标-一站式IT培训就业服务领导者 沪ICP备13042190号-4
上海海同信息科技有限公司 Copyright ©2015 www.zhizuobiao.com,All Rights Reserved.
 沪公网安备 31011502005948号    

©2015 www.zhizuobiao.com All Rights Reserved

208小时内训课程