位置: 编程技术 - 正文

浅析SQL Server中的执行计划缓存(下)(sql server功能介绍)

编辑:rootadmin

推荐整理分享浅析SQL Server中的执行计划缓存(下)(sql server功能介绍),希望有所帮助,仅作参考,欢迎阅读内容。

文章相关热门搜索词:sql server的相关技术知识,sql-server,sql-server,sql server干嘛的,sql server用处,sql server的介绍,sqlserver的go,sqlserver的go,内容如对您有帮助,希望把文章链接给更多的朋友!

在上篇文章给大家介绍了SQL Server中的执行计划缓存(上),本文继续给大家介绍sqlserver执行计划缓存相关知识,小伙伴们一起学习吧。

简介

在上篇文章中我们谈到了查询优化器和执行计划缓存的关系,以及其二者之间的冲突。本篇文章中,我们会主要阐述执行计划缓存常见的问题以及一些解决办法。

将执行缓存考虑在内时的流程

上篇文章中提到了查询优化器解析语句的过程,当将计划缓存考虑在内时,首先需要查看计划缓存中是否已经有语句的缓存,如果没有,才会执行编译过程,如果存在则直接利用编译好的执行计划。因此,完整的过程如图1所示。

图1.将计划缓存考虑在内的过程

图1中我们可以看到,其中有一步需要在缓存中找到计划的过程。因此不难猜出,只要是这一类查找,一定跑不了散列(Hash)的数据结构。通过sys.dm_os_memory_cache_hash_tables这个DMV可以找到有关该Hash表的一些信息,如图2所示。这里值得注意的是,当执行计划过多导致散列后的对象在同一个Bucket过多时,则需要额外的Bucket,因此可能会导致查找计划缓存效率低下。解决办法是尽量减少在计划缓存中的计划个数,我们会在本文后面讨论到。

图2.有关存储计划缓存的HashTable的相关信息

当出现这类问题时,我们可以在buckets_avg_scan_miss_length列看出问题。这类情况在缓存命中率(SQL Server: Plan Cache-Cache Hit Ratio)比较高,但编译时间过长时可以作为考虑对象。

参数化和非参数化

查询计划的唯一标识是查询语句本身,但假设语句的主体一样,而仅仅是查询条件谓词不一样,那在执行计划中算1个执行计划还是两个执行计划呢?It's Depends。

假设下面两个语句,如图3所示。

图3.仅仅谓词条件不一样的两个语句

虽然执行计划一样,但是在执行计划缓存中却会保留两份执行计划,如图4所示。

图4.同一个语句,不同条件,有两份不同的执行计划缓存

我们知道,执行计划缓存依靠查询语句本身来判别缓存,因此上面两个语句在执行计划缓存中就被视为两个不同的语句。那么解决该问题的手段就是使得执行计划缓存中的查询语句一模一样。

参数化

使得仅仅是某些参数不同,而查询本身相同的语句可以复用,就是参数化的意义所在。比如说图3中的语句,如果我们启用了数据库的强制参数化,或是使用存储过程等。SQL Server会将这些语句强制参数话,比如说我们根据图5修改了数据库层级的选项。

图5.数据库层级的选项

此时我们再来执行图3中的两条语句,通过查询执行计划缓存,我们发现变量部分被参数化了,从而在计划缓存中的语句变得一致,如图6所示,从而可以复用.

图6.参数话之后的查询语句

但是,强制参数会引起一些问题,查询优化器很多时候就无法根据统计信息最优化一些具体的查询,比如说不能应用一些索引或者该扫描的时候却查找。所产生的负面影响在上篇文章中已经说过,这里就不细说了。

因此对于上面的问题可以有几种解决办法。

浅析SQL Server中的执行计划缓存(下)(sql server功能介绍)

平衡参数化和非参数化

在具体的情况下,参数化有些时候是好的,但有些时候却是性能问题的罪魁祸首,下面我们来看几种平衡这两者之间关系的手段。

使用RECOMPILE

当查询中,不准确的执行计划的成本要高于编译的成本时,在存储过程中使用RECOMPILE选项或是在即席查询中使用RECOMPILE提示使得每次查询都会重新生成执行计划,该参数会使得生成的执行计划不会被插入到执行计划缓存中。对于OLAP类查询来说,不准确的执行计划所耗费的成本往往高于编译成本太多,所以可以考虑该参数或选项,您可以如代码清单1中的查询所示这样使用Hint。

代码清单1.使用Recompile

除去我们可以手动提示SQL Server重编译之外,SQL Server也会在下列条件下自动重编译:

元数据变更,比如说表明称改变、删除列、变更数据类型等。统计信息变更。

连接的SET参数变化,SET ANSI_NULLS等的值不一样,会导致缓存的执行计划不能被复用,从而重编译。这也是为什么我们看到缓存的执行计划中语句一模一样,但就是不复用,还需要相关的参数一致,这些参数可以通过sys.dm_exec_plan_attributes来查看。

使用Optimize For参数

RECOMPILE方式提供了完全不使用计划缓存的节奏。但有些时候,特性谓语的执行计划被使用的次数h更多,比如说,仅仅那些谓语条件产生大量返回结果集的参数编译,我们可以考虑Optimize For参数。比如我们来看代码清单2。

代码清单2.使用OPTIMIZE FOR提示

使用了该参数会使得缓存的执行计划按照OPTIMIZE FOR后面的谓语条件来生成并缓存执行计划,这也可能造成不在该参数中的查询效率低下,但是该参数是我们选择的,因此通常我们知道哪些谓语条件会被使用的多一些。

另外,自SQL Server 开始多了一个OPTIMIZE FOR UNKNOWN参数,这使得在优化查询的过程中探测作为谓语条件的局部参数的值,而不是根据局部变量的初始值去探测统计信息。

在存储过程中使用局部变量代替存储过程参数

在存储过程中不使用过程参数,而是使用局部变量相当于直接禁用参数嗅探。毕竟,局部变量的值只有在运行时才能知道,在执行计划被查询优化器编译时是无法知道该值的,因此强迫查询分析器使用条件列的平均值进行估计。

虽然这种方式使得参数估计变得非常不准确,但是会变得非常稳定,毕竟统计信息不会变更的过于频繁。该方式不被推荐,如果可能,尽量使用Optimizer的方式。

代码清单3展示了这种方式。

代码清单3.直接引用局部变量,而不是存储过程参数

强制参数化

在本篇文章的前面已经提到过了强制参数化,这里就不再提了。

使用计划指导

在某些情况下,我们的环境不允许我们直接修改SQL语句,比如所不希望破坏代码的逻辑性或是应用程序是第三方开发,因此无论是加HINT或参数都变得不现实。此时我们可以使用计划指导。

计划指导使得查询语句在由客户端应用程序扔到SQL Server的时候,SQL Server对其加上提示或选项,比如说通过代码清单4可以看到一个计划指导的例子。

代码清单4.对我们前面的查询设置计划指导

当加入了计划指导后,当批处理到达SQL Server时,在查找匹配的计划缓存时也会去找是否有计划指导和其相匹配。如果匹配,则应用计划指导中的提示或选项。这里要注意的是,这里@stmt参数必须和查询语句中的一句一模一样,差一个空格都会被认为不匹配。

PARAMETERIZATION SIMPLE

当我们在数据库层级启用了强制参数化时,对于特定语句,我们却不想启用强制参数化,我们可以使用PARAMETERIZATION SIMPLE选项,如代码清单5所示。

代码清单5.通过计划指南对单条语句应用简单参数化

小结

执行计划缓存希望尽量重用执行计划,这会减少编译所消耗的CPU和执行缓存所消耗的内存。而查询优化器希望尽量生成更精准的执行计划,这势必会造成大量的执行计划,这不仅仅可能引起重编译大量消耗CPU,还会造成内存压力,甚至当执行计划缓存过多超过BUCKET的限制时,在缓存中匹配执行计划的步骤也会消耗更多的时间。

因此利用本篇文章中所述的方法基于实际的情况平衡两者之间的关系,就变得非常重要。

SQL Server中的RAND函数的介绍和区间随机数值函数的实现 工作中会遇到SQLServer模拟数据生成以及数值列值(如整型、日期和时间数据类型)随机填充等等任务,这些任务中都要使用到随机数。鉴于此,本文将

SQL Server FileStream详解 FILESTREAM是SQLServer中的一个新特性,允许以独立文件的形式存放大对象数据,而不是以往一样将所有数据都保存到数据文件中。以往在对业务系统的文

SQL Server中数据行批量插入脚本的存储实现 无意中看到朋友写的一篇文章将表里的数据批量生成INSERT语句的存储过程的实现。我仔细看文中的两个存储代码,自我感觉两个都不太满意,都是生成

标签: sql server功能介绍

本文链接地址:https://www.jiuchutong.com/biancheng/312861.html 转载请保留说明!

上一篇:浅析SQL Server中的执行计划缓存(上)(简述sql server)

下一篇:SQL Server中的RAND函数的介绍和区间随机数值函数的实现(sql随机函数rand怎么用)

  • 建筑业异地预缴税款时间
  • 从租计征房产税低于从价计征房产税
  • 广告公司 印刷
  • 小规模差额征税全额开票和差额开票
  • 8.会计核算方法具体包括哪些内容?
  • 劳务公司收到劳务费怎么做账
  • 预提费用在资产负债表哪个科目
  • 应付利润借方有余额怎么处理
  • 小型制造型企业有哪些
  • 长期投资换入固定资产如何处理
  • 技术转让收入纳税调整
  • 房地产老项目增值税简易计税
  • 个人账户作为公司私账
  • 企业销售现金折扣
  • 货样广告品出口需要开票吗
  • 增值税申报收入和企业所得税申报收入
  • 已经缴纳税款怎么作废申报
  • 委托加工怎么计算消费税
  • 购进旅客运输服务为什么不能抵扣进项税额
  • 预支差旅费的会计科目
  • 同城提入业务包括
  • 高新技术企业如何节税
  • 取得海运费发票怎么入账
  • 内外账合并步骤
  • 进项税税额抵扣
  • 财务软件上线需注意什么
  • win10打开txt
  • 顺丰充值的钱可以退吗
  • 若依项目制作饼状图和柱状图
  • 图像质量评价指标及方法 图像工程
  • 人工智能专业计算机科学与技术
  • 现金流量表中如何考虑折旧费
  • uniapp面试题必问的坑
  • 基于深度学习的车型识别系统(Python+清新界面+数据集)
  • php计时函数
  • python如何提取字典中的键
  • 资产证券化会计处理案例
  • 精灵图的使用
  • 可供分配利润的会计分录
  • 三代手续费支付申请表怎么填
  • 增值税专用发票的税率是多少啊
  • 金税盘注册码怎么查
  • 甲企业持有乙企业40%的股权,能够对乙企业
  • 办公桌椅入什么会计科目
  • 增值税专用发票有几联?
  • 为什么增值税不计入成本费用
  • 赠送客户的商品需要交税吗
  • 票面利率大于市场利率,则发行价格一定( )面值
  • 如何理解什么是半殖民地半封建社会
  • 投资收益科目在利润表中怎么填
  • 注册资本金印花税税率是2.5还是5
  • 以前年度损益调整结转到哪里
  • 购买500元的保险会计分录
  • 合伙人工资薪金
  • 营业收入净额怎么算出来
  • Windows server 2003 最后一代cpu
  • sql server触发器实验
  • 电脑开机一直显示windowsxp
  • Windows 2003 SP2 简体中文版下载地址
  • macos终端删除文件
  • mac中的快捷键大全
  • mac上itunes
  • win10文件资源
  • xshell使用ssh命令远程连接linux
  • fdb是什么文件
  • linux服务器的mac地址怎么查
  • uiview和calayer关系
  • opengl缓冲区
  • perl字符串替换 ~s
  • nodejs获取当前路径
  • [置顶]bilinovel
  • js jquery区别
  • python的爬虫技术
  • unity3d的作用
  • javascript基础教学
  • python 自定义类模块
  • python爬取前程无忧
  • 事业单位大额资金拨付需要什么报账材料
  • 青岛税务局局长是什么级别?
  • 济南市土地使用税申报表怎样作废
  • 免责声明:网站部分图片文字素材来源于网络,如有侵权,请及时告知,我们会第一时间删除,谢谢! 邮箱:opceo@qq.com

    鄂ICP备2023003026号

    网站地图: 企业信息 工商信息 财税知识 网络常识 编程技术

    友情链接: 武汉网站建设