《数据结构》—— 遍历线索二叉树

左手的ㄟ右手 2022-08-28 07:50 312阅读 0赞

遍历线索二叉树

    • 一、中序线索二叉树的遍历查找:DGBEAFC
    • 二、先序线索二叉树的遍历查找:ABDGECF
    • 三、后序线索二叉树的遍历查找:GDEBFCA
    • 四、算法:遍历中序线索二叉树

在这里插入图片描述
以上图为参考:

  • 中序遍历(左根右):DGBEAFC;
  • 先序遍历(根左右):ABDGECF;
  • 后序遍历(左右根):GDEBFCA;

讨论如何在线索二叉树查找结点的前驱与后继?

一、中序线索二叉树的遍历查找:DGBEAFC

① 查找p指针所指结点的前驱:

  • 若 p->Ltag==1,则p的左链指向其前驱;
  • 若 p->Ltag==0,说明有左子树,结点的前驱是遍历左子树时最后访问的一个结点(左子树中最右下结点)。

例如:左根右 —> ((左根右)根右) —> (((左根右)根右)根右);
我们对根结点A进行中序遍历,可以发现,它的左子树是B为根节点,按中序遍历结果是先返回 D-G-B-E-A 满足(左子树最后访问的一个结点)。

② 查找p指针所指结点的后继:

  • 若 p->Rtag==1,则p的左链指向其前驱;
  • 若 p->Rtag==0,说明有右子树,结点的后继是遍历右子树时第一个访问的结点(右子树中最左下结点)。

例如:左根右 —> (左根(左根右));
我们对根结点A进行中序遍历,可以发现,它的右子树是C为根节点,第一个访问是F结点,满足中序遍历结果。
在这里插入图片描述


二、先序线索二叉树的遍历查找:ABDGECF

① 查找p指针所指结点的前驱:

  • 若 p->Ltag==1,则p的左链指向其前驱;
  • 若 p->Ltag==0,说明有左子树,若 *p 结点为根结点,则找不到它的前驱,因为由于(根左右),推不出它的前驱;若 *p 结点为左孩子 || (右孩子&兄弟节点)为NULL,则 *p 结点的前驱是它的父节点; 否则若 *p 结点为右孩子,它的前驱是兄弟节点。

在这里插入图片描述
② 查找p指针所指结点的后继:

  • 若 p->Rtag==1,则p的左链指向其前驱;
  • 若 p->Rtag==0,说明有右子树,由遍历规则知则先序后继为左孩子,若p没有左孩子,则先序后继为右孩子。

在这里插入图片描述


三、后序线索二叉树的遍历查找:GDEBFCA

① 查找p指针所指结点的前驱:

  • 若 p->Ltag==1,则p的左链指向其前驱;
  • 若 p->Ltag==0,若p有右孩子,则后序前驱为右孩子,若p没有右孩子,则后序前驱为左孩子。

② 查找p指针所指结点的后继:

  • 若 p为根节点,则没有后继;若p为左孩子,它的后继结点是它的兄弟结点或者是p的父节点;若p为右孩子,它的后继是父节点。

在这里插入图片描述
在这里插入图片描述

四、算法:遍历中序线索二叉树

  1. #include<iostream>
  2. using namespace std;
  3. //二叉树的二叉线索类型存储表示
  4. typedef struct BiThrNode{
  5. char data;
  6. struct BiThrNode *lchild,*rchild; //左右孩子指针
  7. int LTag,RTag;
  8. }BiThrNode,*BiThrTree;
  9. BiThrNode *pre=new BiThrNode; // 全局变量pre
  10. void CreateBiTree(BiThrTree &T){
  11. //按先序次序输入二叉树中结点的值(一个字符),创建二叉链表表示的二叉树T
  12. char ch;
  13. cin >> ch;
  14. if(ch=='#') T=NULL; //递归结束,建空树
  15. else{
  16. T=new BiThrNode;
  17. T->data=ch;//生成根结点
  18. CreateBiTree(T->lchild); //递归创建左子树
  19. CreateBiTree(T->rchild); //递归创建右子树
  20. }
  21. }
  22. // 结点P为根的子树中序线索化
  23. void InThreading(BiThrTree p){
  24. //pre是全局变量,初始化时其右孩子指针为空,便于在树的最左点开始建线索
  25. if(p){
  26. InThreading(p->lchild); //左子树递归线索化
  27. if(!p->lchild){
  28. //p的左孩子为空
  29. p->LTag=1; //给p加上左线索
  30. p->lchild=pre; //p的左孩子指针指向pre(前驱)
  31. }
  32. else
  33. p->LTag=0;
  34. if(!pre->rchild){
  35. //pre的右孩子为空
  36. pre->RTag=1; //给pre加上右线索
  37. pre->rchild=p; //pre的右孩子指针指向p(后继)
  38. }
  39. else
  40. pre->RTag=0;
  41. pre=p; //保持pre指向p的前驱
  42. InThreading(p->rchild); //右子树递归线索化
  43. }
  44. }
  45. // 带头结点的中序线索化
  46. void InOrderThreading (BiThrTree &Thrt,BiThrTree T){
  47. //中序遍历二叉树T,并将其中序线索化,Thrt指向头结点
  48. Thrt=new BiThrNode; //建头结点
  49. Thrt->LTag=0; //头结点有左孩子,若树非空,则其左孩子为树根
  50. Thrt->RTag=1; //头结点的右孩子指针为右线索
  51. Thrt->rchild=Thrt; //初始化时右指针指向自己
  52. if(!T) Thrt->lchild=Thrt; //若树为空,则左指针也指向自己
  53. else{
  54. Thrt->lchild=T; pre=Thrt; //头结点的左孩子指向根,pre初值指向头结点
  55. InThreading(T); //调用算法,对以T为根的二叉树进行中序线索化
  56. pre->rchild=Thrt; //算法,结束后,pre为最右结点,pre的右线索指向头结点
  57. pre->RTag=1;
  58. Thrt->rchild=pre; //头结点的右线索指向pre
  59. }
  60. }
  61. void InOrderTraverse_Thr(BiThrTree T){
  62. //T指向头结点,头结点的左链lchild指向根结点,可参见线索化算法5.8。
  63. //中序遍历二叉线索树T的非递归算法,对每个数据元素直接输出
  64. BiThrTree p;
  65. p=T->lchild; //p指向根结点
  66. while(p!=T){
  67. //空树或遍历结束时,p==T
  68. while(p->LTag==0) //沿左孩子向下
  69. p=p->lchild; //访问其左子树为空的结点
  70. cout<<p->data;
  71. while(p->RTag==1&&p->rchild!=T){
  72. p=p->rchild; //沿右线索访问后继结点
  73. cout<<p->data;
  74. }
  75. p=p->rchild;
  76. }
  77. }
  78. int main(){
  79. pre->RTag=1;
  80. pre->rchild=NULL;
  81. BiThrTree tree,Thrt;
  82. cout<<"请输入建立二叉链表的序列,例如(ABD#G##E##CF###):\n";
  83. CreateBiTree(tree);
  84. InOrderThreading(Thrt,tree);
  85. cout<<"中序遍历线索二叉树的结果为:\n";
  86. InOrderTraverse_Thr(Thrt);
  87. cout<<endl;
  88. }

在这里插入图片描述

发表评论

表情:
评论列表 (有 0 条评论,312人围观)

还没有评论,来说两句吧...

相关阅读