查找算法 二叉排序树.doc_第1页
查找算法 二叉排序树.doc_第2页
查找算法 二叉排序树.doc_第3页
查找算法 二叉排序树.doc_第4页
查找算法 二叉排序树.doc_第5页
免费预览已结束,剩余3页可下载查看

下载本文档

版权说明:本文档由用户提供并上传,收益归属内容提供方,若内容存在侵权,请进行举报或认领

文档简介

转载 查找算法 二叉排序树转载查找算法-二叉排序树2007-09-21 15:08当用线性表作为表的组织形式时,可以有三种查找法。其中以二分查找效率最高。但由于二分查找要求表中结点按关键字有序,且不能用链表作存储结构,因此,当表的插入或删除操作频繁时,为维护表的有序性,势必要移动表中很多结点。这种由移动结点引起的额外时间开销,就会抵消二分查找的优点。也就是说,二分查找只适用于静态查找表。若要对动态查找表进行高效率的查找,可采用下面介绍的几种特殊的二叉树或树作为表的组织形式。不妨将它们统称为树表。下面将分别讨论在这些树表上进行查找和修改操作的方法。二叉排序树1、二叉排序树的定义二叉排序树(Binary Sort Tree)又称二叉查找(搜索)树(Binary Search Tree)。其定义为:二叉排序树或者是空树,或者是满足如下性质的二叉树:若它的左子树非空,则左子树上所有结点的值均小于根结点的值;若它的右子树非空,则右子树上所有结点的值均大于根结点的值;左、右子树本身又各是一棵二叉排序树。上述性质简称二叉排序树性质(BST性质),故二叉排序树实际上是满足BST性质的二叉树。2、二叉排序树的特点由BST性质可得:(1)二叉排序树中任一结点x,其左(右)子树中任一结点y(若存在)的关键字必小(大)于x的关键字。(2)二叉排序树中,各结点关键字是惟一的。注意:实际应用中,不能保证被查找的数据集中各元素的关键字互不相同,所以可将二叉排序树定义中BST性质(1)里的小于改为大于等于,或将BST性质(2)里的大于改为小于等于,甚至可同时修改这两个性质。(3)按中序遍历该树所得到的中序序列是一个递增有序序列。【例】下图所示的两棵树均是二叉排序树,它们的中序序列均为有序序列:2,3,4,5,7,8。3、二叉排序树的存储结构typedef int KeyType;/假定关键字类型为整数typedef struct node/结点类型KeyType key;/关键字项InfoType otherinfo;/其它数据域,InfoType视应用情况而定,下面不处理它struct node*lchild,*rchild;/左右孩子指针BSTNode;typedef BSTNode*BSTree;/BSTree是二叉排序树的类型4、二叉排序树上的运算(1)二叉排序树的插入和生成二叉排序树插入新结点的过程在二叉排序树中插入新结点,要保证插入后仍满足BST性质。其插入过程是:(a)若二叉排序树T为空,则为待插入的关键字key申请一个新结点,并令其为根;(b)若二叉排序树T不为空,则将key和根的关键字比较:(i)若二者相等,则说明树中已有此关键字key,无须插入。(ii)若key Tkey,则将key插入根的左子树中。(iii)若key Tkey,则将它插入根的右子树中。子树中的插入过程与上述的树中插入过程相同。如此进行下去,直到将key作为一个新的叶结点的关键字插入到二叉排序树中,或者直到发现树中已有此关键字为止。二叉排序树插入新结点的递归算法【参见参考书目】二叉排序树插入新结点的非递归算法void InsertBST(BSTree*Tptr,KeyType key)/若二叉排序树*Tptr中没有关键字为key,则插入,否则直接返回BSTNode*f,*p=*TPtr;/p的初值指向根结点while(p)/查找插入位置if(p-key=key)return;/树中已有key,无须插入f=p;/f保存当前查找的结点p=(key p-key)?p-lchild:p-rchild;/若key p-key,则在左子树中查找,否则在右子树中查找/endwhile p=(BSTNode*)malloc(sizeof(BSTNode);p-key=key;p-lchild=p-rchild=NULL;/生成新结点if(*TPtr=NULL)/原树为空*Tptr=p;/新插入的结点为新的根else/原树非空时将新结点关p作为关f的左孩子或右孩子插入if(key f-key)f-lchild=p;else f-rchild=p;/InsertBST二叉排序树的生成二叉排序树的生成,是从空的二叉排序树开始,每输入一个结点数据,就调用一次插入算法将它插入到当前已生成的二叉排序树中。生成二叉排序树的算法如下:BSTree CreateBST(void)/输入一个结点序列,建立一棵二叉排序树,将根结点指针返回BSTree T=NULL;/初始时T为空树KeyType key;scanf(%d,&key);/读人一个关键字while(key)/假设key=0是输人结束标志InsertBST(&T,key);/将key插入二叉排序树T scanf(%d,&key);/读人下一关键字return T;/返回建立的二叉排序树的根指针/BSTree二叉排序树的生成过程由输入实例(5,3,7,2,4,8),根据生成二叉排序树算法生成二叉排序树的过程【参见动画演示】注意:输入序列决定了二叉排序树的形态。二叉排序树的中序序列是一个有序序列。所以对于一个任意的关键字序列构造一棵二叉排序树,其实质是对此关键字序列进行排序,使其变为有序序列。排序树的名称也由此而来。通常将这种排序称为树排序(Tree Sort),可以证明这种排序的平均执行时间亦为O(nlgn)。对相同的输入实例,树排序的执行时间约为堆排序的2至3倍。因此在一般情况下,构造二叉排序树的目的并非为了排序,而是用它来加速查找,这是因为在一个有序的集合上查找通常比在无序集合上查找更快。因此,人们又常常将二叉排序树称为二叉查找树。(2)二叉排序树的删除从二叉排序树中删除一个结点,不能把以该结点为根的子树都删去,并且还要保证删除后所得的二叉树仍然满足BST性质。删除操作的一般步骤(1)进行查找查找时,令p指向当前访问到的结点,parent指向其双亲(其初值为NULL)。若树中找不到被删结点则返回,否则被删结点是*p。(2)删去*p。删*p时,应将*p的子树(若有)仍连接在树上且保持BST性质不变。按*p的孩子数目分三种情况进行处理。删除*p结点的三种情况(1)*p是叶子(即它的孩子数为0)无须连接*p的子树,只需将*p的双亲*parent中指向*p的指针域置空即可。(2)*p只有一个孩子*child只需将*child和*p的双亲直接连接后,即可删去*p。注意:*p既可能是*parent的左孩子也可能是其右孩子,而*child可能是*p的左孩子或右孩子,故共有4种状态,具体【参见动画演示】。(3)*p有两个孩子先令q=p,将被删结点的地址保存在q中;然后找*q的中序后继*p,并在查找过程中仍用parent记住*p的双亲位置。*q的中序后继*p一定是*q的右子树中最左下的结点,它无左子树。因此,可以将删去*q的操作转换为删去的*p的操作,即在释放结点*p之前将其数据复制到*q中,就相当于删去了*q。具体【参见动画演示】。二叉排序树删除算法分析:上述三种情况都能统一到情况(2),算法中只需针对情况(2)处理即可。注意边界条件:若parent为空,被删结点*p是根,故删去*p后,应将child置为根。算法:void DelBSTNode(BSTree*Tptr,KeyType key)/在二叉排序树*Tptr中删去关键字为key的结点BSTNode*parent=NUll,*p=*Tptr,*q,*child;while(p)/从根开始查找关键字为key的待删结点if(p-key=key)break;/已找到,跳出查找循环parent=p;/parent指向*p的双亲p=(key p-key)?p-lchild:p-rchild;/在关p的左或右子树中继续找if(!p)return;/找不到被删结点则返回q=p;/q记住被删结点*p if(q-lchild&q-rchild)/*q的两个孩子均非空,故找*q的中序后继*p for(parent=q,p=q-rchild;p-lchild;parent=p,p=p=-lchild);/现在情况(3)已被转换为情况(2),而情况(1)相当于是情况(2)中child=NULL的状况child=(p-lchild)?p-lchild:p-rchild;/若是情况(2),则child非空;否则child为空if(!parent)/*p的双亲为空,说明*p为根,删*p后应修改根指针*Tptr=child;/若是情况(1),则删去*p后,树为空;否则child变为根else/*p不是根,将*p的孩子和*p的双亲进行连接,*p从树上被摘下if(p=parent-lchild)/*p是双亲的左孩子parent-lchild=child;/*child作为*parent的左孩子else parent-rchild=child;/*child作为parent的右孩子if(p!=q)/是情况(3),需将*p的数据复制到*q q-key=p-key;/若还有其它数据域亦需复制/endif free(p);/释放*p占用的空间/DelBSTNode二叉排序树的删除运算实例具体参见【动画演示】(3)二叉排序树上的查找查找递归算法在二叉排序树上进行查找,和二分查找类似,也是一个逐步缩小查找范围的过程。递归的查找算法:BSTNode*SearchBST(BSTree T,KeyType key)/在二叉排序树T上查找关键字为key的结点,成功时返回该结点位置,否则返回NUll if(T=NULL|key=T-key)/递归的终结条件return T;/T为空,查找失败;否则成功,返回找到的结点位置if(key T-key)return SearchBST(T-lchild,key);else return SearchBST(T-rchild,key);/继续在右子树中查找/SearchBST算法分析在二叉排序树上进行查找时,若查找成功,则是从根结点出发走了一条从根到待查结点的路径。若查找不成功,则是从根结点出发走了一条从根到某个叶子的路径。(1)二叉排序树查找成功的平均查找长度在等概率假设下,下面(a)图中二叉排序树查找成功的平均查找长度为在等概率假设下,(b)图所示的树在查找成功时的平均查找长度为:ASLb=(1+2+3+4+5+6+7+8+9+10)/10=5.5注意:与二分查找类似,和关键字比较的次数不超过树的深度。(2)在二叉排序树上进行查找时的平均查找长度和二叉树的形态有关二分查找法查找长度为n的有序表,其判定树是惟一的。含有n个结点的二叉排序树却不惟一。对于含有同样一组结点的表,由于结点插入的先后次序不同,所构成的二叉排序树的形态和深度也可能不同【例】下图(a)所示的树,是按如下插入次序构成的:45,24,55,12,37,53,60,28,40,70下图(b)所示的树,是按如下插入次序构成的:12,24,28,37,40,45,53,55,60,70在二叉排序树上进行查找时的平均查找长度和二叉树的形态有关:在最坏情况下,二叉排序树是通过把一个有序表的n个结点依次插入而生成的,此时所得的二叉排序树蜕化为棵深度为n的单支树,它的平均查找长度和单链表上的顺序查找相同,亦是(n+1)/2。在最好情况下,二叉排序树在生成的过程中,树的形态比较匀称,最终得到的是一棵形态与二分查找的判定树相似的二叉排序树,此时它的平均查找长度大约是lgn。插入、删除和查找算法的时间复杂度均为O(lgn)。(3)二叉排序树和二分查找的比较就平均时间性能而言,二叉排序树上的查找和二分查找差不多。就维护表的有序性而言,二叉排序树无须移动结点,只需修改指针即可完成插入和删除操作,且其平均的执行时间均为O(lgn),因此更有效。二分查找所涉及的有序表是一个向量,若有插入和删除结点的操作,则维护表的有序性所花的代价是O(n)。当有序表是静态查找表时,宜用向量作为其存储结构,而采用二分查找实现其查找操作;若有序表里动态查找表,则应选择二叉排序树作为其存储结构。(4)平衡二叉树为了保证二叉排序树的高度为lgn,从而保证然二叉排序树上实现的插入、删除和查找等基本操作的平均时间为O(lgn),在往树中插入或删除结点时,要调整树的形态来保持树的平衡。使之既保持BST性质不变又保证树

温馨提示

  • 1. 本站所有资源如无特殊说明,都需要本地电脑安装OFFICE2007和PDF阅读器。图纸软件为CAD,CAXA,PROE,UG,SolidWorks等.压缩文件请下载最新的WinRAR软件解压。
  • 2. 本站的文档不包含任何第三方提供的附件图纸等,如果需要附件,请联系上传者。文件的所有权益归上传用户所有。
  • 3. 本站RAR压缩包中若带图纸,网页内容里面会有图纸预览,若没有图纸预览就没有图纸。
  • 4. 未经权益所有人同意不得将文件中的内容挪作商业或盈利用途。
  • 5. 人人文库网仅提供信息存储空间,仅对用户上传内容的表现方式做保护处理,对用户上传分享的文档内容本身不做任何修改或编辑,并不能对任何下载内容负责。
  • 6. 下载文件中如有侵权或不适当内容,请与我们联系,我们立即纠正。
  • 7. 本站不保证下载资源的准确性、安全性和完整性, 同时也不承担用户因使用这些下载资源对自己和他人造成任何形式的伤害或损失。

评论

0/150

提交评论