当前位置:   article > 正文

详解二叉排序树_二叉排序树的构造过程

二叉排序树的构造过程

二叉排序树的插入、查找、删除

二叉排序树的定义

二叉排序树右称二叉查找树。或者为空树,或者是具有以下性质:

(1)若它的左子树不为空,则左子树所有节点的值小于根结点,

(2)若它的右子树不为空,则根结点的值小于所有右子树结点的值

(3)它的左右子树叶分别为二叉排序树

 

总结起来就是根据结点的值有:左子树<根结点<右子树

如下图就是一棵二叉排序树

 

 

它的中序遍历:12、19、23、28、34、36、38、42、53、65、90,刚好是排好序的。

 

二叉排序树的查找分析

二叉排序树有一些基本的应用,可以用于查找,查找的长度与树的深度有关。其平均查找长度为logN(以2位底的对数,N为结点个数)。

 

一个序列的排序二叉树有多种构造情况。下面考虑两种极端情况:深度最小和深度最大的二叉排序树。

关键字序列(45,24,53,12,37,93),假设6个记录的查找概率相等都为1/6

 

 

容易看出这是一棵完全二叉树,该数的深度(最大层次)为3,它的平均查找长度为:

ASL = 1/6(1+2+2+3+3+3)= 14/6

 

再来看该关键字序列构成深度最大的二叉排序树

 

 

这是一棵单支树,其深度为6,它的平均查找长度为:

ASL = 1/6(1+2+3+4+5+6)= 21/6

 

在随机情况下,该数的平均查找长度总是介于这两种情况之间,即14/6  < log6 < 21/6。二叉排序树的平均查找长度和logN是等数量级的。

 

二叉排序树的构造过程

以上述完全叉树序列为例,构造该二叉排序数的过程如下图:

 

 

容易看出,每次插入新的结点都是二叉排序树上的叶子结点,则在插入时,不必移动其他结点,只需改动某个结点的指针,由空变为非空即可。

 

结点的插入操作与二叉排序树的定义紧密相关,即左<根<右,新插入一个关键字时,从根结点开始比较,直到找到合适的插入位置为止。还有一种情况就是一个序列中可能有两个相同的关键字,对于这种情况,向树中插入关键字时遇到相同关键字时,什么都不做,不进行重复插入操作。

 

二叉排序树的删除过程

删除二叉排序树中任意一个结点,需要分成以下三种情况讨论:

(1)删除的是叶子结点

这种情况最简单,由于删除叶子结点不会破坏整棵树的结构,只需要修改其双亲结点的指针即可。

(2)删除的结点只有左子树或只有右子树。

这种情况只需要将其左子树或右子树往上推,替代要删除结点的位置,显然,作此修改也不会破坏二叉排序树的结构。

(3)删除的结点左右子树都不为空

这种情况稍微复杂一点,为了不保持二叉排序树的结构,有两种思路,一种就是从要删除结点的左子树中选取最大的结点进行替代之,另一种就是从要删除的结点的右子树中选取最小的结点替代之。

下面考虑第三种情况

 

f、p、q、s分别为指向结点F、P、Q、S的指针,假如现要删除结点P,它的左右子树都不为空。

其一就是寻找P的左子树中的最大结点,代替之。

易知左子树中最大结点为S,将P结点用S结点代替,还需处理S结点的左孩子K,由于S只有左子树K,在删除结点S之后,只要令K为S的双亲Q的右子树即可。

 

其二就是寻找P结点的右子树中最小的结点,代替之。

找到P后,第一步就是往右走一步到R,若R的左子树不为空,则最小的结点在R的左子树中,只需一直往左走即p=p->Left,指定p->Left为空(右子树最小结点,其左子树定为空

,则找到最小结点,将其代替要删除的结点P,并将最小结点的右子树代替该最小结点即可。如下图,Z为要删除结点中右子树中最小结点,将P用Z代替后,Z的位置再由Y代替

 

 

若R的左子树为空,那么最小结点就是R,只需将R往上推代替P

具体实现:

 

  1. /*BinarySearchTree.cpp*/
  2. #include<stdio.h>
  3. #include<stdlib.h>
  4. typedef int ElementType;
  5. typedef struct TreeNode{
  6. ElementType Element;
  7. struct TreeNode *Left;
  8. struct TreeNode *Right;
  9. }TreeNode,*SearchTree;
  10. SearchTree Insert(SearchTree T,ElementType e)
  11. {
  12. //二叉排序树的插入过程
  13. if(!T)
  14. {
  15. //未找到值为e的结点,就新生成一个结点
  16. T = (TreeNode*)malloc(sizeof(TreeNode));
  17. T->Element = e;
  18. T->Left = NULL;
  19. T->Right = NULL;
  20. }
  21. else if(e<T->Element) //在左子树中插入
  22. T->Left = Insert(T->Left,e);
  23. else if(e>T->Element) //在右子树中插入
  24. T->Right = Insert(T->Right,e);
  25. else //e == T->Element do nothing
  26. {
  27. }
  28. return T;
  29. }
  30. int Delete(SearchTree &p)
  31. {
  32. //从二叉排序树中删除结点p,并重接它的左子树或右子树
  33. TreeNode *q,*s;
  34. if(!p->Right) //右子树为空,重接左子树
  35. {
  36. q = p;
  37. p = p->Left;
  38. free(q);
  39. }
  40. else if(!p->Left) //左子树为空,重接右子树
  41. {
  42. q = p;
  43. p = p->Right;
  44. free(q);
  45. }
  46. else //左右子树都不为空
  47. {
  48. q = p;
  49. s = p->Left;
  50. while(s->Right)
  51. {
  52. q = s; //q是s的前驱结点
  53. s = s->Right; //往右走到尽头
  54. }
  55. p->Element = s->Element;
  56. if(q != p)
  57. q->Right = s->Left;
  58. else //q = p;
  59. q->Left = s->Left;
  60. free(s);
  61. }
  62. return 1;
  63. }
  64. int DeleteBST(SearchTree &T,ElementType x)//考虑这里为什么用引用?
  65. {
  66. //在排序二叉树T中删除关键字等于x的结点
  67. if(!T) //不存在关键字等于x的数据元素
  68. return -1;
  69. else
  70. {
  71. if(x == T->Element) //查找成功
  72. return Delete(T);
  73. else if(x < T->Element)
  74. return DeleteBST(T->Left,x); //在左子树中继续查找
  75. else
  76. return DeleteBST(T->Right,x); //在右子树中继续查找
  77. }
  78. return 1;
  79. }
  80. void InOderTravesal(SearchTree T)
  81. {
  82. //中序遍历
  83. if(T)
  84. {
  85. InOderTravesal(T->Left);
  86. printf("%d ",T->Element);
  87. InOderTravesal(T->Right);
  88. }
  89. }
  90. void removeTree(SearchTree T)
  91. {
  92. //销毁二叉树
  93. if(T)
  94. {
  95. removeTree(T->Left);
  96. removeTree(T->Right);
  97. free(T);
  98. T=NULL;
  99. }
  100. }
  101. int main()
  102. {
  103. SearchTree T=NULL;
  104. int a[7] = {45,24,53,45,12,24,90},i;
  105. for(i=0; i<7; ++i)
  106. T=Insert(T,a[i]);
  107. InOderTravesal(T);
  108. printf("\n");
  109. DeleteBST(T,24);
  110. InOderTravesal(T);
  111. removeTree(T);
  112. return 0;
  113. }

 

 

声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/w/不正经/article/detail/709658
推荐阅读
相关标签
  

闽ICP备14008679号