本文属于《理解性能的奥秘——应用程序中慢,SSMS中快》系列

接上文:理解性能的奥秘——应用程序中慢,SSMS中快(2)——SQL Server如何编译存储过程
在我们开始深入研究如何处理参数嗅探相关的性能问题之前,由于这个课题过于广泛,所以首先先介绍一些跟参数嗅探没有直接关系的内容,但是又会导致语句在SSMS和应用程序中存在性能差异的情况。

替换变量和参数:

前面已经接触过,但是在这里对其进行扩展。有时会看到论坛上有人说,某个存储过程很慢,但是把相同的语句提取出来单独执行就很快。真相就是:语句涉及了变量,可能是本地变量或者参数。为了单独检查语句问题,他们把变量替换成常量。但是前面提到过,单独的语句和在存储过程中很不一样,当用常量替换变量时,SQL Server可以更加准确地预估影响行数,从而生成更好的查询计划。并且SQL Server不用考虑下一次运行是,常量会被改变。
另外一个类似的错误就是把参数放入变量中,如:

CREATE PROCEDURE some_sp @par1 int 
AS
   ...
   -- 某些使用到变量 @par1的语句

你想检查某个语句问题,所以你改成:
DECLARE @par1 int
SELECT @par1 = 4711
-- 接下来的语句

上一节已经说过,这种方式和原来方式有很大不同,SQL Server因为不清楚本地变量的情况,所以会使用标准假设(一般就是表总数的30%)。但是如果你有一个千行代码的存储过程,然后其中某个语句很慢,要如何定位问题,同时又能保持与在存储过程中一样的环境呢?其中一个简单但也存在限制的方法就是添加“OPTION(RECOMPILE)”提示到语句的最后,这个提示仅在SQL 2005中有效,可以触发语句对所有参数、变量在运行时进行参数嗅探和重编译。
但是这个方法只对只有一个参数的查询或者存储过程中没有本地变量的情况有用。(因为本地变量的值通常不会被嗅探。)另外,这个方法对SQL 2008及后续版本无效,因为OPTION(RECOMPILE)的实现方式变得更加合理:如果所有变量值都是常数,SQL Server会编译查询。(由于一个致命bug的存在,微软不得不把这种行为恢复,所以并不能应用于所有的SQL 2008版本中)。
一个总能生效的方式就是把语句封装到sp_executesql中:

EXEC sp_executesql N'-- 牵涉到参数的语句 @par1', N'@par1 int', 4711
这里需要注意的是,对于字符文本,需要使用双引号包住。如果查询涉及本地变量,可以在动态SQL中赋值。
还有一种方式就是创建一个带有问题语句的虚拟存储过程,为了避免数据库中存在垃圾对象,你可以使用临时存储过程:
CREATE PROCEDURE #test @par1 int AS
   -- 问题语句.

如果是动态SQL ,也同样确保在这类存储过程中进行局部定义本地变量。但是需要警告一下,目前还没发现SQL Server是否有对临时存储过程有特殊的调整或优化的限制。

阻塞:


不要忘记应用程序运行慢的其中一个可能原因就是与阻塞 有关。当你在3小时后在SSMS中测试查询时,阻塞源可能已经结束。如果你在SSMS中不管是否改动ARITHABORT,运行存储过程都没问题,一直都很快,那么阻塞就应该纳入考虑内容并进行深究。但是这个话题很大,也超出了本系列的主旨,所以可以看作者的另外一篇文章(以后有空也会一起翻译)beta lockinfo


索引视图和索引化的计算列:

在SQL 2000时代,这个问题比后续版本严重很多,从SQL Server 2005开始,所以视图和索引化后的计算列(包括SQL 2008加入的过滤索引filter index)在语句编译期间,下面的设置必须为ON:QUOTED_IDENTIFIER, ANSI_NULLS, ANSI_WARNINGS, ANSI_PADDING, CONCAT_NULL_YIELDS_NUL,而NUMERIC_ROUNDABORT必须为OFF。

而在SQL 2000中,应用程序使用默认SET选项对索引化的计算列和索引视图是没有什么好处的。但是即使存在参数嗅探,当你在SSMS或查询分析器中运行是,性能也往往可能好很多,因为ARITHABORT默认为ON。

但是由于SQL 2000时代过去太久了,估计已经很少人还在用,所以如果对这个内容有兴趣,读者可以去看原文,因为这部分主要是对SQL 2000描述的。点击打开链接



链接服务器的问题:

这里假定你用的远程服务器是SQL 2012 SP1之前的版本。假设下面的语句:
SELECT C.*
FROM SOME_SERVER.Northwind.dbo.Orders O
JOIN Customers C
	ON O.CustomerID = C.CustomerID
WHERE O.OrderID > 20000

以两个不同的用户运行了语句两次。第一个用户是两台服务器的sysadmin,第二个用户仅有SELECT权限。为了确保得到不同缓存条目, 使用了不同的ARITHABORT设置。在以sysadmin运行时,查询计划如下:



以普通用户运行是,得到的执行计划如下:

为什么查询计划不一样呢?可以肯定的是不是参数嗅探问题,因为语句没有参数。但是当发现查询计划不是与其形状或者预期操作符时,应该看看预估行数。这两个查询计划的“Remote Query”操作符的属性如下:左边的是第一个,右边的是第二个


可以发现预估行数不一样。当以sysadmin身份运行时,预估行数为1,这是正确的数量,因为Northwind库的Orders表中不存在Order ID超过20000的数据。但是当以普通用户运行时,预估行数为249行。也就是表总数的30%。因为此时统计信息丢失导致优化器对数据量的预估不准确。
当查询直接访问本地实例的表时,优化器可以得到语句相关的所有表的统计信息,不存在额外的权限检查。 当SQL Server访问一个链接服务器时,服务器之间的通讯并没有使用专用的的协议,而是使用标准的OLE DB接口访问链接服务器,其他诸如sql server实例、oracle、文本文件或任何自定义数据源也是如此。具体如何获取统计信息还要取决于数据源和请求的OLE DB Provider。 在这种情况下,SQL Server Native Client会通过两步获得统计信息(可以在远程服务器上使用Profiler检查):
  1. SQL Server Native Client驱动会先运行sp_table_statistics2_rowset返回统计信息包含的列的信息,也包括了基数信息和密度信息。
  2. 驱动运行DBCC SHOW_STATISTICS,返回完整的分布统计信息。
直到从SQL Server 2012 RTM位置,为了有足够权限运行DBCC SHWO_STATISTICS命令,必须使用属于sysadmin角色或者数据库层面的db_owner或db_ddladmin角色的权限。
所以使用不同权限的账号运行同一个语句的时候,会得到不同的结果,当以sysadmin运行的时候,可以获取完整的分布信息,意味着能发现没有数据复合orderid>20000的条件,所以预估行数为1(注意SQL Server永远不会用预估行数为0代表没有匹配数据)。但是以普通用户运行时,因为权限问题导致DBCC SHOW_STATISTICS失败,而且不抛出错误,以优化器接受“没有统计信息”为替代方案,然后进行默认假设。所以在基数预估的时候就不够准确了。
不管什么时候,当你遇到一个包含链接服务器查询,在应用程序中很慢,在SSMS中很快时,你都应该检查一下是否权限问题导致的。如果是权限问题,那么可以考虑下面方案:
  • 可以把用户加到远程数据库的db_ddladmin角色中,但是这个意味着用户可以增加和删除表,一般不建议。
  • 默认情况下,一个用户连接远程服务器的时候,都是使用远程服务器中相同的账号,但是可以使用sp_addlinkedsrvlogin映射 一个登录号,以便用户有权限执行属于db_ddladminde 角色的事情。但是这个账号必须是SQL 登录账号,所以要确保远程服务器是否启用了混合身份验证,但是这种方案从安全角度来说还是有问题。
  • 某些情况下可以改写成OPENQUERY强制在远程服务器中进行预估。如果查询包含了多个远程表的情况下,这种方式特别有效。(但是也有风险,因为优化器可能只能从远程服务器中获取更少的统计信息。)
  • 也可以使用完整的提示和计划向导来实现你期望的查询计划。
  • 最后,要考虑一下是否真的有必要使用链接服务器而不能放在同一个 服务器上?能否修改?或者有其他解决方案?
再次提醒:重要的是查询在远程服务器上是否有权限,而不是在本地服务器上。另外,这个问题仅存在于远程服务器的SQL Server版本是SQL Server 2012 RTM及以前版本的SQL Server。从SQL 2012 SP1开始,执行DBCC SHOW_STATISTICS的权限要求已经被放宽,只要有SELECT权限即可执行。

小结:

本节没有介绍参数嗅探的情况,而是 介绍一些同样也会导致在应用程序中很慢,在SSMS中很快的情况。但是因为本主题主要是介绍参数嗅探,所以还是尽可能把篇幅放在这个点上,下一节将会介绍:
  • 收集解决参数嗅探问题的信息




理解性能的奥秘——应用程序中慢,SSMS中快(3)——不总是参数嗅探的错的更多相关文章

  1. 理解性能的奥秘——应用程序中慢,SSMS中快(6)——SQL Server如何编译动态SQL

    本文属于<理解性能的奥秘--应用程序中慢,SSMS中快>系列 接上文:理解性能的奥秘--应用程序中慢,SSMS中快(5)--案例:如何应对参数嗅探 我们抛开参数嗅探的话题,回到了本系列的最 ...

  2. 理解性能的奥秘——应用程序中慢,SSMS中快(5)——案例:如何应对参数嗅探

    本文属于<理解性能的奥秘--应用程序中慢,SSMS中快>系列 接上文:理解性能的奥秘--应用程序中慢,SSMS中快(4)--收集解决参数嗅探问题的信息 首先我们需要明白,参数嗅探本身不是问 ...

  3. 理解性能的奥秘——应用程序中慢,SSMS中快(4)——收集解决参数嗅探问题的信息

    本文属于<理解性能的奥秘--应用程序中慢,SSMS中快>系列 接上文:理解性能的奥秘--应用程序中慢,SSMS中快(3)--不总是参数嗅探的错 前面已经提到过关于存储过程在SSMS中运行很 ...

  4. 理解性能的奥秘——应用程序中慢,SSMS中快(2)——SQL Server如何编译存储过程

    本文属于<理解性能的奥秘--应用程序中慢,SSMS中快>系列 接上文:理解性能的奥秘--应用程序中慢,SSMS中快(1)--简介 本文介绍SQL Server如何编译存储过程并使用计划缓存 ...

  5. 理解性能的奥秘——应用程序中慢,SSMS中快(1)——简介

    本文属于<理解性能的奥秘--应用程序中慢,SSMS中快>系列 在工作中发现有不少类似的现象,有幸看到国外大牛写的一篇文章,由于已经完善得不能再添油加醋,所以决定直接翻译,原文出处:http ...

  6. 理解性能的奥秘——应用程序中慢,SSMS中快(4)收集解决参数嗅探问题的信息

    ---从计划缓存中直接获取查询计划和参数: ), ) SELECT @dbname = 'hydee_连锁', @procname = 'dbo.p_select_ware'; WITH baseda ...

  7. [转]提高 Linux 上 socket 性能,加速网络应用程序的 4 种方法

    原文链接:http://www.ibm.com/developerworks/cn/linux/l-hisock.html 使用 Sockets API,我们可以开发客户机和服务器应用程序,它们可以在 ...

  8. 1   开发一个注重性能的JDBC应用程序不是一件容易的事. 当你的代码运行很慢的时候JDBC驱动程序并不会抛出异常告诉你。   本系列的性能提示将为改善JDBC应用程序的性能介绍一些基本的指导原则,这其中的原则已经被许多现有的JDBC应用程序编译运行并验证过。 这些指导原则包括:    正确的使用数据库MetaData方法    只获取需要的数据    选用最佳性能的功能    管理连

    1 开发一个注重性能的JDBC应用程序不是一件容易的事. 当你的代码运行很慢的时候JDBC驱动程序并不会抛出异常告诉你. 本系列的性能提示将为改善JDBC应用程序的性能介绍一些基本的指导原则,这其中的 ...

  9. 【SQL server初级】数据库性能优化三:程序操作优化

    数据库优化包含以下三部分,数据库自身的优化,数据库表优化,程序操作优化.此文为第三部分 数据库性能优化三:程序操作优化 概述:程序访问优化也可以认为是访问SQL语句的优化,一个好的SQL语句是可以减少 ...

随机推荐

  1. LAMP+Wordpress搭建博客

    作为一名技术控,看到别人都有专属于自己的博客,小夜也忍不住自己操刀做个自己的博客.有兴趣的童鞋接下来可要认真地练习比对哦.如果出现错误也不要着急上火,咱们可以交流学习.我们的口号是:遇水搭桥,遇山凿山 ...

  2. [LeetCode] Find Pivot Index 寻找中枢点

    Given an array of integers nums, write a method that returns the "pivot" index of this arr ...

  3. 【webstorm使用手册】如何安装插件

    Files->Settings, 搜索"plugin",如图,

  4. PyQt5 QSerialPort子线程操作

    环境: python3.6 pyqt5 只是简单的一个思路,请忽略脆弱的异常防护: # -*- coding: utf-8 -*- import sys from PyQt5.QtWidgets im ...

  5. 【JDK1.8】JUC——LockSupport

    一.前言 Basic thread blocking primitives for creating locks and other synchronization classes. 用于创建锁定和其 ...

  6. [SDOI 2013]方程

    Description 题库链接 求不定方程 \(x_1+x_2+\cdots +x_n=m\) 的正整数解的个数,并且要求满足限定: \(\forall i\in[1,n_1] x_i\leq a_ ...

  7. NOIP 2009 最优贸易

    题目描述 C 国有 n 个大城市和 m 条道路,每条道路连接这 n 个城市中的某两个城市.任意两个 城市之间最多只有一条道路直接相连.这 m 条道路中有一部分为单向通行的道路,一部分 为双向通行的道路 ...

  8. 【BZOJ1036】【ZJOI2008】数的统计

    接着找树剖的题...传送门(点我) 题意:给你一棵无根树,有三种操作:查询树上2点路径的点权和/最大点权:更改某点的点权. 解题思路:树链剖分裸题,我采用了常数较小的zkw线段树维护剖下来的树(毕竟线 ...

  9. bzoj省选十连测推广赛

    A.普通计算姬 题意:给丁一棵树,每个点有一个权值,用sum(x)表示以x为根的子树的权值和,要求支持两种操作: 1 u v  :修改点u的权值为v. 2 l  r   :  求∑sum[i] l&l ...

  10. n个并发进程共用一个公共变量Q,写出用信号灯实现n个进程互斥的程序描述,给出信号灯值得取值范围,并说明每个取值范围的物理意义。

    答: var mutex: semaphore:=1; begin cobegin process i : begin   // i = 1,2,……,n repeat P(mutex); 对公共变量 ...