`

Oracle中的Hash Join详解

阅读更多
一、  hash join概念

     hash join(HJ)是一种用于equi-join(而anti-join就是使用NOT IN时的join)的技术。在Oracle中,它是从7.3开始引入的,
以代替sort-merge和nested-loop join方式,提高效率。在CBO(hash join只有在CBO才可能被使用到)模式下,优化器计算代价时,
首先会考虑hash join。
     可以通过提示use_hash来强制使用hash join,也可以通过修改会话或数据库参数HASH_JOIN_ENABLED=FALSE(默认为TRUE)强
制不使用hash join。
     Hash join的主要资源消耗在于CPU(在内存中创建临时的hash表,并进行hash计算),而merge join的资源消耗主要在于此盘IO
(扫描表或索引)。在并行系统中,hash join对CPU的消耗更加明显。所以在CPU紧张时,最好限制使用hash join。
     在绝大多数情况下,hash join效率比其他join方式效率更高:
     在Sort-Merge Join(SMJ),两张表的数据都需要先做排序,然后做merge。因此效率相对最差;
     Nested-Loop Join(NL)效率比SMJ更高。特别是当驱动表的数据量很大(集的势高)时。这样可以并行扫描内表。
     Hash join效率最高,因为只要对两张表扫描一次。

     Hash join一般用于一张小表和一张大表进行join时。Hash join的过程大致如下(下面所说的内存就指sort area,关于过程,后
面会作详细讨论):
1.  一张小表被hash在内存中。因为数据量小,所以这张小表的大多数数据已经驻入在内存中,剩下的少量数据被放置在临时表空间中;
2.  每读取大表的一条记录,就和小表中内存中的数据进行比较,如果符合,则立即输出数据(也就是说没有读取临时表空间中的小表的数
据)。而如果大表的数据与小表中临时表空间的数据相符合,则不直接输出,而是也被存储临时表空间中。
3.  当大表的所有数据都读取完毕,将临时表空间中的数据以其输出。

     如果小表的数据量足够小(小于hash area size),那所有数据就都在内存中了,可以避免对临时表空间的读写。

     如果是并行环境下,前面中的第2步就变成如下了:
2.  每读取一条大表的记录,和内存中小表的数据比较,如果符合先做join,而不直接输出,直到整张大表数据读取完毕。如果内存足够,
Join好的数据就保存在内存中。否则,就保存在临时表空间中。
二、  Oracle中与hash join相关的参数

     首先,要注意的是,hash join只有在CBO方式下才会被激活。在oracle中与hash join相关的参数主要有以下几个:
1.             HASH_JOIN_ENABLED

     这个参数是控制查询计划是否采用hash join的“总开关”。它可以在会话级和实例级被修改。默认为TRUE,既可以(不是一定,要看优
化器计算出来的代价)使用。如果设为FALSE,则禁止使用hash join。
2.             HASH_AREA_SIZE

     这个参数控制每个会话的hash内存空间有多大。它也可以在会话级和实例级被修改。默认(也是推荐)值是sort area空间大小的两倍
(2*SORT_AREA_SIZE)。要提高hash join的效率,就一定尽量保证sort area足够大,能容纳下整个小表的数据。但是因为每个会话都会
开辟一个这么大的内存空间作为hash内存,所以不能过大(一般不建议超过2M)。
     在Oracle9i及以后版本中,Oracle不推荐在dedicated server中使用这个参数来设置hash内存,而是推荐通过设置
PGA_AGGRATE_TARGET参数来自动管理PGA内存。保留HASH_AREA_SIZE只是为了向后兼容。在dedicated server中,hash area是从
PGA中分配的,而在MTS(Multi-Threaded Server)中,hash area是从UGA中分配的。
     另外,还要注意的是,每个会话并不一定只打开一个hash area,因为一个查询中可能不止一个hash join,这是就会相应同时打开多个
hash area。
3.             HAHS_MULTIBLOCK_IO_COUNT

     这个参数决定每次读入hash area的数据块数量。因此它会对IO性能产生影响。他只能在init.ora或spfile中修改。在8.0及之前版本,
它的默认值是1,在8i及以后版本,默认值是0。一般设置为1-(65536/DB_BLOCK_SIZE)。
     在9i中,这个参数是一个隐藏参数:_HASH_MULTIBLOCK_IO_COUNT,可以通过表x$ksppi查询和修改。
     另外,在MTS中,这个参数将不起作用(只会使用1)。
     它的最大值受到OS的IO带宽和DB_BLOCK_SIZE的影响。既不能大于MAX_IO_SIZE/DB_BLOCK_SIZE。
     在8i及以后版本,如果这个值设置为0,则表示在每次查询时,Oracle自己自动计算这个值。这个值对IO性能影响非常大,因此,建议不要
修改这个参数,使用默认值0,让Oracle自己去计算这个值。
     如果一定要设置这个值,要保证以下不等式能成立:
     R/M < Po2(M/C)
     其中,R表示小表的大小;M=HASH_AREA_SIZE*0.9;Po2(n)为n的2次方;C=HASH_MULTIBLOCK_IO_COUNT*DB_BLOCK_SIZE。
三、  Hash join的过程

     一次完整的hash join如下:
1.             计算小表的分区(bucket)数

     决定hash join的一个重要因素是小表的分区(bucket)数。这个数字由hash_area_size、hash_multiblock_io_count和
db_block_size参数共同决定。Oracle会保留hash area的20%来存储分区的头信息、hash位图信息和hash表。因此,这个数字的计算公式是:
     Bucket数=0.8*hash_area_size/(hash_multiblock_io_count*db_block_size)
2.             Hash计算  

     读取小表数据(简称为R),并对每一条数据根据hash算法进行计算。Oracle采用两种hash算法进行计算,计算出能达到最快速度的hash值
(第一hash值和第二hash值)。而关于这些分区的全部hash值(第一hash值)就成为hash表。
3.             存放数据到hash内存中

     将经过hash算法计算的数据,根据各个bucket的hash值(第一hash值)分别放入相应的bucket中。第二hash值就存放在各条记录中。
4.             创建hash位图

     与此同时,也创建了一个关于这两个hash值映射关系的hash位图。
5.             超出内存大小部分被移到磁盘

     如果hash area被占满,那最大一个分区就会被写到磁盘(临时表空间)上去。任何需要写入到磁盘分区上的记录都会导致磁盘分区被更新。这
样的话,就会严重影响性能,因此一定要尽量避免这种情况。
     2-5一直持续到整个表的数据读取完毕。
6.             对分区排序

     为了能充分利用内存,尽量存储更多的分区,Oracle会按照各个分区的大小将他们在内存中排序。
7.             读取大表数据,进行hash匹配

     接下来就开始读取大表(简称S)中的数据。按顺序每读取一条记录,计算它的hash值,并检查是否与内存中的分区的hash值一致。如果是,返
回join数据。如果内存中的分区没有符合的,就将S中的数据写入到一个新的分区中,这个分区也采用与计算R一样的算法计算出hash值。也就是说这些
S中的数据产生的新的分区数应该和R的分区集的分区数一样。这些新的分区被存储在磁盘(临时表空间)上。
8.             完全大表全部数据的读取

     一直按照7进行,直到大表中的所有数据的读取完毕。

9.             处理没有join的数据

     这个时候就产生了一大堆join好的数据和从R和S中计算存储在磁盘上的分区。
10.       二次hash计算

     从R和S的分区集中抽取出最小的一个分区,使用第二种hash函数计算出并在内存中创建hash表。采用第二种hash函数的原因是为了使数据分布
性更好。
11.       二次hash匹配

     在从另一个数据源(与hash在内存的那个分区所属数据源不同的)中读取分区数据,与内存中的新hash表进行匹配。返回join数据。
12.       完成全部hash join

     继续按照9-11处理剩余分区,直到全部处理完毕。

     整个hash join就完成了。

四、  关于唯一健值的hash位图

     这个位图包含了每个hash分区是否有有值的信息。它记录了有数据的分区的hash值。这个位图的最大作用就是,如果S表中的数据没有与内存中的
hash表匹配上,先查看这个位图,已决定是否将没有匹配的数据写入磁盘。那些不可能匹配到的数据(即位图上对应的分区没有数据)就不再写入磁盘。

转自:http://www.hellodba.com/reader.php?ID=144&lang=cn
分享到:
评论

相关推荐

    分布式协调服务Zookeeper的安装配置与基本操作解析

    内容概要:本文档提供了一份详细的Zookeeper从下载、安装、配置到启动的操作指南。重点介绍了如何在三台Linux系统上安装配置Zookeeper集群以及如何使用Zookeeper客户端的基本命令完成常见任务,如创建/更新/删除/查询节点等。 适合人群:对分布式系统管理有兴趣的技术人员,尤其是需要使用或维护基于Zookeeper的应用系统的开发者。 使用场景及目标:适用于准备搭建高可用性和稳定性的分布式应用系统的企业或者团队,在遇到节点状态同步、集群管理和数据发布等问题时,可利用Zookeeper作为解决方案来构建更为强大的基础设施。 其他说明:本指南对于新手入门级教程,可以帮助没有经验的用户快速上手部署和操作Zookeeper,但进阶功能需查阅更多资料深入研究。

    Android开发:ADB:ADB高级调试技术.pdf

    Android开发:ADB:ADB高级调试技术.pdf

    三菱plc实例程序66个,制糖机、送板机、收板机、卫生巾小包机、压入机、弯管机、橡胶注射机、橡胶硫化机、细针自动排序机等等

    万盼制糖机器.rar 下是自己写的QD75程式现场使用过OK.rar 物品移置程序机械手程序.rar 窑炉传动控制程序.rar 线网自动焊接专机程序.rar 线路板廠钢板磨刷水洗烘干机.rar 线路板自动叠合拆解线.rar 细针自动排序机.rar 维达吹瓶机多段温度PID控制 亚智收板机.rar 协易高精密冲床部份程序.rar 卧式裁断机.rar 卫生巾小包机.rar 卫生巾机,涂布机等涂胶高速补偿控制程序(正在用).rar 卫生巾设备.rar 卫生巾设备三菱PLC程序外加HMI触屏程序QCPU有注解.zip 卫生级灭菌柜三菱PLC程序及人机GP程序.rar 压入机.rar 压机.rar 压机程序.rar 压花机3.rar 压铸机.rar 压铸机周边程序.rar 小型水电站.rar 小日本的程序.rar 小水电站.rar 小车定位计数程序.zip 弯管机程序.rar 弯管机触摸屏+plc程序(全套三菱)带注解.rar 往复刀追膜070518.rar 循环流化床锅炉的10组共40个吹灰器程序.rar 摇盘机伺服程序.rar 新亿成染色电脑与三菱通讯程序.rar 新印刷机.rar

    dy一键典藏评工具.zip

    dy一键典藏评工具.zip

    基于VB+ACCESS 实现的汽车美容管理系统(源代码+系统+外文翻译+英文文献+开题报告+任务书)

    【作品名称】:基于VB+ACCESS 实现的汽车美容管理系统(源代码+系统+外文翻译+英文文献+开题报告+任务书) 【适用人群】:适用于希望学习不同技术领域的小白或进阶学习者。可作为毕设项目、课程设计、大作业、工程实训或初期项目立项。 【项目介绍】: 为改善企业中因手工统计车辆、材料、人事、财务等各类信息的烦琐性及复杂性,在节省企业工作时间的同时提高企业的工作效率,达到提升企业效益的目的,因此急需一套完整的计算机化的汽车美容店信息管理解决方案。 开发的这套高效率的计算机信息管理系统解决了当前汽车美容店业务流程管理难度大、头绪多、耗费大量人力和财力的问题,充分发挥了计算机容量大、运算速度快、精度高的优点。 此软件以Access 2000作为后台数据库,在Visual Basic 6.0平台下开发的,需要在Windows操作系统下运行。 关键词:信息管理,数据库,财务,统计,Visual Basic 6.0 【资源声明】:本资源作为“参考资料”而不是“定制需求”,代码只能作为参考,不能完全复制照搬。需要有一定的基础看懂代码,自行调试代码并解决报错,能自行添加功能修改代码。

    关于电机中绕组出现大小线圈时嵌线方式的探讨

    关于电机中绕组出现大小线圈时嵌线方式的探讨

    GoC编程基础:魔法学院的奇幻之旅-从零开始学习C++画图

    内容概要:本教程旨在带领孩子们从零开始学习GoC(C++画图)编程,涵盖基础编程概念、画图命令、色彩变换、图形组合等多种技能点。每节课通过具体实例讲解编程知识,并提供练习题帮助巩固所学内容。整个教程分为15课,涉及初级、中级到高级的各种图形绘制技巧。 适合人群:小学至初中阶段的学生,对编程感兴趣的孩子及家长,希望学习编程基础知识的教学人员。 使用场景及目标:用于学校的编程兴趣班、课外辅导班或家庭自学,培养孩子的编程兴趣和基本编程能力,提升逻辑思维和解决问题的能力。 其他说明:课程内容丰富多样,涵盖了从简单到复杂的图形绘制,适合不同层次的学习者逐步提升技能。配套资源包括详细的课程讲解视频和丰富的练习题库,有助于学员全面掌握GoC编程的基础知识。

    西南交通大学-数据结构实验1(2022级)

    数据结构实验中的实验内容大部分由我亲自书写,参考教材为严蔚敏第二版。大部分代码与教材一致。数据结构实验第一次独自开始做课程实验,因此代码风格从实验1到最后的课程设计一直在变化,希望可以谅解。有部分实验内容由于时间仓促,可能会有bug,在使用时如有bug,请在评论区或者私信留言,我会在空闲时间更改。

    最终幻想7 重制版种子 最火的3A游戏大作

    这就是那个90G的最终幻想7高清3D重制版 3A大作!!!重量级宝藏,积分不够的私信我。

    【雷达信号分选】基于matlab CDIF算法雷达信号分选仿真【含Matlab源码 8945期】.mp4

    Matlab领域上传的视频均有对应的完整代码,皆可运行,亲测可用,适合小白; 1、代码压缩包内容 主函数:main.m; 调用函数:其他m文件;无需运行 运行结果效果图; 2、代码运行版本 Matlab 2019b;若运行有误,根据提示修改;若不会,私信博主; 3、运行操作步骤 步骤一:将所有文件放到Matlab的当前文件夹中; 步骤二:双击打开main.m文件; 步骤三:点击运行,等程序运行完得到结果; 4、仿真咨询 如需其他服务,可私信博主; 4.1 博客或资源的完整代码提供 4.2 期刊或参考文献复现 4.3 Matlab程序定制 4.4 科研合作

    介绍点云开发的基础知识

    介绍点云开发的基础知识

    JAVA五子棋手机网络对战游戏的设计与实现(源代码+).zip

    JAVA五子棋手机网络对战游戏的设计与实现,文件中包含图片说明、项目源码及使用说明,资料仅供学习使用。

    财务管理系统 JAVA毕业设计 源码+数据库+论文+启动教程(SpringBoot+Vue.JS).zip

    财务管理系统 JAVA毕业设计 源码+数据库+论文+启动教程(SpringBoot+Vue.JS) 项目启动教程:https://www.bilibili.com/video/BV11ktveuE2d

    AB plc程序100个(一).zip,内含100个plc程序源码,.ACD文件,和相关文档

    AB plc程序100个(一),总共588个AB plc程序源码由于太大了分成几个发,这是第一个,此资源内含100个plc程序

    交易明细分析工具3.2.py

    交易明细分析工具3.2.py

    煜寒咨询:2022年奶茶行业报告.pdf

    煜寒咨询:2022年奶茶行业报告.pdf

    AcountMS.rar

    AcountMS.rar

    Android开发:Android Studio:事件处理与监听器.pdf

    Android开发:Android Studio:事件处理与监听器.pdf

    优化建模的 MATLAB 工具箱.zip

    优化建模的 MATLAB 工具箱

    算法与数据结构-综合提升 C++版

    算法与数据结构-综合提升 C++版 资源列表: 00-0pening 01-Why-Algorithms 02-Sorting-Basic 03-Sorting-Advance 04-Heap 05-Binary-Search-Tree 06-Union-Find 07-Graph-Basics08-Minimum-Span-Trees 09-Shortest-Path 10-Ending

Global site tag (gtag.js) - Google Analytics