位置: 编程技术 - 正文

浅谈SQL Server中的三种物理连接操作(性能比较)(sql server的介绍)

编辑:rootadmin
在SQL Server中,我们所常见的表与表之间的Inner Join,Outer Join都会被执行引擎根据所选的列,数据上是否有索引,所选数据的选择性转化为Loop Join,Merge Join,Hash Join这三种物理连接中的一种。理解这三种物理连接是理解在表连接时解决性能问题的基础,下面我来对这三种连接的原理,适用场景进行描述。 嵌套循环连接(Nested Loop Join) 循环嵌套连接是最基本的连接,正如其名所示那样,需要进行循环嵌套,嵌套循环是三种方式中唯一支持不等式连接的方式,这种连接方式的过程可以简单的用下图展示: 图1.循环嵌套连接的第一步

图2.循环嵌套连接的第二步

由上面两个图不难看出,循环嵌套连接查找内部循环表的次数等于外部循环的行数,当外部循环没有更多的行时,循环嵌套结束。另外,还可以看出,这种连接方式需要内部循环的表有序(也就是有索引),并且外部循环表的行数要小于内部循环的行数,否则查询分析器就更倾向于Hash Join(会在本文后面讲到)。

通过嵌套循环连接也可以看出,随着数据量的增长这种方式对性能的消耗将呈现出指数级别的增长,所以数据量到一定程度时,查询分析器往往就会采用这种方式。

下面我们通过例子来看一下循环嵌套连接,利用微软的AdventureWorks数据库:

图3.一个简单的嵌套循环连接

图3中ProductID是有索引的,并且在循环的外部表中(Product表)符合ProductID=的行有条,因此,对应的SalesOrderDetail表需要查找次。让我们在上面的查询中再考虑另外一个例子,如图4所示。

图4.额外的列带来的额外的书签查找

由图4中可以看出,由于多选择了一个UnitPrice列,导致了连接的索引无法覆盖所求查询,必须通过书签查找来进行,这也是为什么我们要养成只Select需要的列的好习惯,为了解决上面的问题,我们既可以用覆盖索引,也可以减少所需的列来避免书签查找。另外,上面符合ProductID的行仅仅只有5条,所以查询分析器会选择书签查找,假如我们将符合条件的行进行增大,查询分析器会倾向于表扫描(通常来说达到表中行数的1%以上往往就会进行table scan而不是书签查找,但这并不绝对),如图5所示。

图5.查询分析器选择了表扫描

可以看出,查询分析器此时选择了表扫描来进行连接,这种方式效率要低下很多,因此好的覆盖索引和Select *都是需要注意的地方。另外,上面情况即使涉及到表扫描,依然是比较理想的情况,更糟糕的情况是使用多个不等式作为连接时,查询分析器即使知道每一个列的统计分布,但却不知道几个条件的联合分布,从而产生错误的执行计划,如图6所示。

图6.由于无法预估联合分布,导致的偏差

由图6中,我们可以看出,估计的行数和实际的行数存在巨大的偏差,从而应该使用表扫描但查询分析器选择了书签查找,这种情况对性能的影响将会比表扫描更加巨大。具体大到什么程度呢?我们可以通过强制表扫描和查询分析器的默认计划进行比对,如图7所示。

图7.强制表扫描性能反而更好

合并连接(Merge Join)

谈到合并连接,我突然想起在西雅图参加SQL Pass峰会晚上酒吧排队点酒,由于我和另外一哥们站错了位置,貌似我们两个在插队一样,我赶紧说:I'm sorry,i thought here is end of line。对方无不幽默的说:”It's OK,In SQL Server,We called it merge join”。

由上面的小故事不难看出,Merge Join其实上就是将两个有序队列进行连接,需要两端都已经有序,所以不必像Loop Join那样不断的查找循环内部的表。其次,Merge Join需要表连接条件中至少有一个等号查询分析器才会去选择Merge Join。

Merge Join的过程我们可以简单用下面图进行描述:

图8.Merge Join第一步

Merge Join首先从两个输入集合中各取第一行,如果匹配,则返回匹配行。加入两行不匹配,则有较小值的输入集合+1,如图9所示。

图9.更小值的输入集合向下进1

用C#代码表示Merge Join的话如代码1所示。

代码1.Merge Join的C#代码表示

因此,通常来说Merge Join如果输入两端有序,则Merge Join效率会非常高,但是如果需要使用显式Sort来保证有序实现Merge Join的话,那么Hash Join将会是效率更高的选择。但是也有一种例外,那就是查询中存在order by,group by,distinct等可能导致查询分析器不得不进行显式排序,那么对于查询分析器来说,反正都已经进行显式Sort了,何不一石二鸟的直接利用Sort后的结果进行成本更小的MERGE JOIN?在这种情况下,Merge Join将会是更好的选择。

另外,我们可以由Merge Join的原理看出,当连接条件为不等式(但不包括!=),比如说> < >=等方式时,Merge Join有着更好的效率。

下面我们来看一个简单的Merge Join,这个Merge Join是由聚集索引和非聚集索引来保证Merge Join的两端有序,如图所示。

图.由聚集索引和非聚集索引保证输入两端有序

当然,当Order By,Group By时查询分析器不得不用显式Sort,从而可以一箭双雕时,也会选择Merge Join而不是Hash Join,如图所示。

图.一箭双雕的Merge Join

哈希匹配(Hash Join)

哈希匹配连接相对前面两种方式更加复杂一些,但是哈希匹配对于大量数据,并且无序的情况下性能均好于Merge Join和Loop Join。对于连接列没有排序的情况下(也就是没有索引),查询分析器会倾向于使用Hash Join。

哈希匹配分为两个阶段,分别为生成和探测阶段,首先是生成阶段,第一阶段生成阶段具体的过程可以如图所示。

图.哈希匹配的第一阶段

图中,将输入源中的每一个条目经过散列函数的计算都放到不同的Hash Bucket中,其中Hash Function的选择和Hash Bucket的数量都是黑盒,微软并没有公布具体的算法,但我相信已经是非常好的算法了。另外在Hash Bucket之内的条目是无序的。通常来讲,查询优化器都会使用连接两端中比较小的哪个输入集来作为第一阶段的输入源。

接下来是探测阶段,对于另一个输入集合,同样针对每一行进行散列函数,确定其所应在的Hash Bucket,在针对这行和对应Hash Bucket中的每一行进行匹配,如果匹配则返回对应的行。

通过了解哈希匹配的原理不难看出,哈希匹配涉及到散列函数,所以对CPU的消耗会非常高,此外,在Hash Bucket中的行是无序的,所以输出结果也是无序的。图是一个典型的哈希匹配,其中查询分析器使用了表数据量比较小的Product表作为生成,而使用数据量大的SalesOrderDetail表作为探测。

图.一个典型的哈希匹配连接

上面的情况都是内存可以容纳下生成阶段所需的内存,如果内存吃紧,则还会涉及到Grace哈希匹配和递归哈希匹配,这就可能会用到TempDB从而吃掉大量的IO。这里就不细说了,有兴趣的同学可以移步: 总结 下面我们通过一个表格简单总结这几种连接方式的消耗和使用场景: 嵌套循环连接合并连接哈希连接适用场景外层循环小,内存循环条件列有序输入两端都有序数据量大,且没有索引CPU低低(如果没有显式排序)高内存低低(如果没有显式排序)高IO可能高可能低低可能高可能低理解SQL Server这几种物理连接方式对于性能调优来说必不可少,很多时候当筛选条件多表连接多时,查询分析器就可能不是那么智能了,因此理解这几种连接方式对于定位问题变得尤为重要。此外,我们也可以通过从业务角度减少查询范围来减少低下性能连接的可能性。 参考文献: 文章来自:

推荐整理分享浅谈SQL Server中的三种物理连接操作(性能比较)(sql server的介绍),希望有所帮助,仅作参考,欢迎阅读内容。

浅谈SQL Server中的三种物理连接操作(性能比较)(sql server的介绍)

文章相关热门搜索词:sql server的sql语句,sql server用处,sql server用处,sql server用处,sql servers,sql server?,sql servers,sql server用处,内容如对您有帮助,希望把文章链接给更多的朋友!

SQL Server误区日谈 第1天 正在运行的事务在服务器故障转移后继续执行 误区#1:在服务器故障转移后,正在运行的事务继续执行这当然是错误的!每次故障转移都伴随着某种形式的恢复。但是如果当正在执行的事务没有Commit

SQL Server误区日谈 第2天 DBCC CHECKDB会导致阻塞 误区#2:DBCCCHECKDB会引起阻塞,因为这个命令默认会加锁这是错误的!在SQLServer7.0以及之前的版本中,DBCCCHECKDB命令的本质是C语言实现的一个不断嵌套循

SQL Server误区日谈 第3天 即时文件初始化特性可以在SQL Server中开启和关闭 本系列文章是我在sqlskill.com的PAUL的博客看到的,很多误区都比较具有典型性和代表性,原文来自T-SQLTuesday#:Misconceptionsabout....EVERYTHING!!,经过我们团队

标签: sql server的介绍

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

上一篇:sqlserver获取各种形式的时间(sql server 获取今天的数据)

下一篇:SQL Server误区30日谈 第1天 正在运行的事务在服务器故障转移后继续执行(sql server错误和使用情况报告)

  • 免征增值税和增值税区别
  • 没有发票的房租如何入账
  • 间接费用分配方法不包括哪些
  • 代订机票被骗可以追回吗
  • 一般纳税人工会经费返还政策
  • 中小企业增值税税率
  • 开发间接费什么时候计提
  • 房贷利息抵个税细则 两套房
  • 新准则 开办费
  • 小规模纳税人计提增值税
  • 支付金融机构手续费计入什么费用
  • 不动产税征收标准
  • 财务报表与汇算报表区别
  • 银行手续费要纳进去吗
  • 简易办法征收一般纳税人开票税率
  • 企业开票代码是什么意思
  • 小配件出口怎么报关
  • 建安类增值税专用发票什么时候改的
  • 小微企业差旅费可以税前扣除吗
  • 2021最新车船税
  • 国税申报纳税调整项目怎么填的?
  • 销售费用广告费存在哪些问题
  • 补贴收入营业外收入比例过高
  • 跨省经营如何缴纳流转税?
  • 外贸公司有出口退税吗
  • 个人所得税款项性质
  • 戴尔r710服务器系统安装详解
  • 如何防止程序被关闭
  • 固定资产残值率可以为0吗
  • mysql-relay-bin
  • 在php中,字符串有哪些表示形式
  • 股权转让协议的注意事项
  • flash是什么文件
  • 没签订采购合同怎么办
  • 外汇返佣金
  • 不锈钢广告牌制作价格
  • 财行〔2019〕11号代扣代缴手续费的解读
  • vue定时调用方法
  • 卸载的应用怎么找回来
  • 造孽啊啥意思
  • 检测费可以抵扣增值税吗
  • 增值税附加税减免政策2022年
  • 买商品赠送赠品怎么做账
  • 企业坏账准备提取的方法和提取的比例由国家统一规定
  • 租金计入主营业务成本吗
  • 结转成本费用
  • 企业所得税预缴可以不交吗
  • sqlserver 通用存储过程分页代码(附使用ROW_NUMBER()和不使用ROW_NUMBER()两种情况性能分析)
  • 无法取得发票的成本能列支吗
  • 公司网站设计怎么做
  • 企业会计本年利润
  • 个税汇算清缴什么意思?
  • 房地产企业销售房屋印花税税率
  • 公司购买土地建厂房账务处理小企业会计准则
  • 本月增值税申报截止日
  • 土地使用税的账务处理分录
  • 分公司交总公司管理费怎么做账
  • 土地收储款是谁给谁
  • 交通费用报销明细表
  • 属于费用类的会计科目有哪些
  • 企业改制后如何管理
  • 最全的sql执行顺序
  • appservicesdkscripterror
  • win7系统怎样
  • win1020h2累积更新
  • windows8飞行模式怎么关
  • 11月 Win8.1 Update 3更新哪些内容?开始菜单依然没有
  • ContentProvider之通过ContentResolver获取图像、视频、音频举例
  • 使用驱动器u盘之前需要格式化
  • jquery使用css
  • python+Django+apache的配置方法详解
  • 科技型中小企业认定有效期
  • 工行网银如何申请发票
  • 安徽省国家税务局网上办税平台
  • 税务总局2017年11号公告第二条
  • 纳税人在国外签字流程
  • 德清社保局固定电话多少
  • 2023浙江高考首考状元
  • 化妆品关税怎么算
  • 中国十大经济农村
  • 免责声明:网站部分图片文字素材来源于网络,如有侵权,请及时告知,我们会第一时间删除,谢谢! 邮箱:opceo@qq.com

    鄂ICP备2023003026号

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

    友情链接: 武汉网站建设