SQL Server里书签查找的性能伤害

发布时间 - 2026-01-11 00:17:59    点击率:

在我的博客上,以前我经常谈到SQL Serverl里的书签查找,还有它们带来的很多问题。在今天的文章里,我想从性能角度进一步谈下书签查找,还有它们如何拉低你整个SQL Server性能。

书签查找——反复循环

如果你的非聚集索引不是个覆盖非聚集索引,SQL Server的查询优化器会引入书签查找。对于从非聚集索引你返回的每一行,SQL Server需要在聚集索引里或堆表里进行额外的查找操作。

例如当你的的聚集索引包含3层,为了返回必要的信息,对于每一行,你需要3页额外的读取。因此,查询优化器再执行计划里选择书签查找操作,仅在有意义的时候发生——基于你查询的选择度。下图展示了有书签查找操作的执行计划。

通常人们不会太关注书签查找,因为它们只执行几次。如果你的查询选择度太低,查询优化器会用聚集索引扫描或表扫描运算符直接扫描整个表。但只在SQL Server重用缓存的执行计划,这个计划是有多次不同运行值,包含书签查找的(基于最初提供的输入值),因此这个情况很容易发生,书签查找反复执行。

为了演示这个性能问题,接下来的查询我指定查询优化器使用特定的非聚集索引。查询本身返回80000行,因为对于每个查询执行,SQL Server需要进行书签查找80000次——反复执行。

CREATE PROCEDURE RetrieveData
AS
 SELECT * FROM Table1 WITH (INDEX(idxTable1_Column2))
 WHERE Column3 = 2
GO

下图展示了查询执行后的实际执行计划。

执行计划看起来非常恐怖(查询优化器甚至启用了并行计划!),因为书签查找运算符这里执行了80000次,查询本身产生了超过165000个逻辑读!(逻辑读个数可以从STATISTIC IO里获取)。

接下来向你展示下,当你有很多并行用户执行这个糟糕查询时,SQL Server会发生什么。我会使用ostress.exe(RML工具的一部分)来模拟100个并行用户的查询。

ostress.exe -Q”EXEC BookmarkLookupsPerformance.dbo.RetrieveData” -n100 -q

在我的测试系统上花费了近15秒来完成100个并行查询。在此期间,CPU占用很高,因为SQL Server需要嵌套循环运算符来进行书签查找操作。嵌套循环操作当然很占CPU资源。

现在让我们修改索引设计,为这个查询创建覆盖非聚集索引。有了非聚集索引,查询优化器不需要再执行计划里进行书签查找。一个非聚集索引查找就可以返回同样的结果:

CREATE NONCLUSTERED INDEX idxTable1_Column2 ON Table1(Column3)
INCLUDE (Column2)
WITH (DROP_EXISTING = ON)
GO

这次当我们再次用ostress.exe执行同个查询,我们看到每个查询在5秒内完成。和我们刚才看到的15秒有很大的区别。这就是覆盖非聚集索引的威力:在我们查询里气门请求的数据都可以在非聚集索引里直接找到,因此书签查找就可以避免。

小结

在这个文章里我向你展示了不好的书签查找会伤及性能。因此,对于重要的查询快速完成查询非常重要——而使用并行的书签查找的执行计划并不是好的选择。这里覆盖非聚集索引可以帮到你。下次设计索引时可以考虑下这个方法。

以上就是本文的全部内容,希望本文的内容对大家的学习或者工作能带来一定的帮助,同时也希望多多支持!


# sql  # server  # 书签查找  # 性能  # Sql Server查询性能优化之不可小觑的书签查找介绍  # 运算符  # 向你  # 展示了  # 我想  # 是个  # 我会  # 在这个  # 是有  # 让我们  # 在此  # 有很多  # 这就是  # 几次  # 当你  # 很容易  # 很高  # 到你  # 就可  # 只在  # 当我们 


相关栏目: 【 网站优化151355 】 【 网络推广146373 】 【 网络技术251813 】 【 AI营销90571


相关推荐: 如何在自有机房高效搭建专业网站?  PHP 实现电台节目表的智能时间匹配与今日/明日轮播逻辑  详解jQuery停止动画——stop()方法的使用  什么是javascript作用域_全局和局部作用域有什么区别?  如何续费美橙建站之星域名及服务?  Laravel Eloquent访问器与修改器是什么_Laravel Accessors & Mutators数据处理技巧  如何为不同团队 ID 动态生成多个“认领值班”按钮  如何快速搭建高效WAP手机网站吸引移动用户?  如何在万网自助建站平台快速创建网站?  Laravel怎么实现微信登录_Laravel Socialite第三方登录集成  做企业网站制作流程,企业网站制作基本流程有哪些?  深圳网站制作的公司有哪些,dido官方网站?  详解jQuery中的事件  Laravel如何使用Livewire构建动态组件?(入门代码)  高防服务器租用首荐平台,企业级优惠套餐快速部署  Laravel如何实现全文搜索功能?(Scout和Algolia示例)  如何在阿里云虚拟服务器快速搭建网站?  如何在 React 中条件性地遍历数组并渲染元素  Laravel怎么实现API接口鉴权_Laravel Sanctum令牌生成与请求验证【教程】  JavaScript如何实现路由_前端路由原理是什么  如何在云主机上快速搭建网站?  详解一款开源免费的.NET文档操作组件DocX(.NET组件介绍之一)  如何自定义建站之星模板颜色并下载新样式?  如何快速打造个性化非模板自助建站?  Laravel如何使用Scope本地作用域_Laravel模型常用查询逻辑封装技巧【手册】  如何快速上传自定义模板至建站之星?  郑州企业网站制作公司,郑州招聘网站有哪些?  如何在建站之星网店版论坛获取技术支持?  Laravel Eloquent性能优化技巧_Laravel N+1查询问题解决  制作无缝贴图网站有哪些,3dmax无缝贴图怎么调?  🚀拖拽式CMS建站能否实现高效与个性化并存?  佛山网站制作系统,佛山企业变更地址网上办理步骤?  Python制作简易注册登录系统  laravel怎么配置和使用PHP-FPM来优化性能_laravel PHP-FPM配置与性能优化方法  如何做网站制作流程,*游戏网站怎么搭建?  如何有效防御Web建站篡改攻击?  Windows10电脑怎么设置虚拟光驱_Win10右键装载ISO镜像文件  Python图片处理进阶教程_Pillow滤镜与图像增强  微信小程序制作网站有哪些,微信小程序需要做网站吗?  Python3.6正式版新特性预览  JS弹性运动实现方法分析  详解Nginx + Tomcat 反向代理 如何在高效的在一台服务器部署多个站点  如何快速搭建二级域名独立网站?  简单实现jsp分页  如何在宝塔面板创建新站点?  Python高阶函数应用_函数作为参数说明【指导】  怎么用AI帮你设计一套个性化的手机App图标?  Laravel怎么进行数据库事务处理_Laravel DB Facade事务操作确保数据一致性  在Oracle关闭情况下如何修改spfile的参数  最好的网站制作公司,网购哪个网站口碑最好,推荐几个?谢谢?