为防止广告,目前nocow只有登录用户能够创建新页面。如要创建页面请先登录/注册(新用户需要等待1个小时才能正常使用该功能)。
线索二叉树
来自NOCOW
线索二叉树(保留遍历时结点在任一序列的前驱和后继的信息):若结点有左子树,则其lchild域指示其左孩子,否则令lchild域指示其前驱;若结点有右子树,则其rchild域指示其右孩子,否则令rchild指示其后继。还需在结点结构中增加两个标志域LTag和RTag。LTag=0时,lchild域指示结点的左孩子,LTag=1时,lchild域指示结点的前驱;RTag=0时,rchild域指示结点的右孩子,RTag=1时,rchild域指示结点的后继。以这种结点结构构成的二叉链表作为二叉树的存储结构,叫做线索链表,其中指向结点前驱和后继的指针叫做线索,加上线索的二叉树称为线索二叉树。对二叉树以某种次序遍历使其变为线索二叉树的过程叫做线索化。若对二叉树进行中序遍历,则所得的线索二叉树称为中序线索二叉树,线索链表称为为中序线索链表。线索二叉树是一种物理结构。
在中序线索树找结点后继的规律是:若其右标志为1,则右链为线索,指示其后继,否则遍历其右子树时访问的第一个结点(右子树最左下的结点)为其后继;找结点前驱的规律是:若其左标志为1,则左链为线索,指示其前驱,否则遍历左子树时最后访问的一个结点(左子树中最右下的结点)为其前驱。在后序线索树中找到结点的后继分三种情况:
- 若结点是二叉树的根,则其后继为空;
- 若结点是其双亲的右孩子,或是其双亲的左孩子且其双亲没有右子树,则其后继即为双亲结点;
- 若结点是其双亲的左孩子,且其双亲有右子树,则其后继为双亲右子树上按后序遍历列出的第一个结点。
[编辑] 二叉线索存储表示
[编辑] 存储结构
二叉树的二叉线索存储表示:在线索链表上添加一个头结点,并令其lchild域的指针指向二叉树的根结点,其rchild域的指针指向中序遍历时访问的最后一个结点。令二叉树中序序列中的第一个结点的lchild域指针和最后一个结点的rchild域的指针均指向头结点,这样就建立了一个双向线索链表
/* 二叉树的二叉线索存储表示 */
typedef enum{Link,Thread}PointerTag; /* Link(0):指针,Thread(1):线索 */
typedef struct BiThrNode
{
TElemType data;
struct BiThrNode *lchild,*rchild; /* 左右孩子指针 */
PointerTag LTag,RTag; /* 左右标志 */
}BiThrNode,*BiThrTree;
[编辑] 基本操作
/* 二叉树的二叉线索存储的基本操作 */
void CreateBiThrTree(BiThrTree *T)
{ /* 按先序输入线索二叉树中结点的值,构造线索二叉树T。0(整型)/空格(字符型)表示空结点 */
TElemType ch;
scanf(form,&ch);
if(ch==Nil)
*T=NULL;
else
{
*T=(BiThrTree)malloc(sizeof(BiThrNode)); /* 生成根结点(先序) */
if(!*T)
exit(OVERFLOW);
(*T)->data=ch; /* 给根结点赋植 */
CreateBiThrTree(&(*T)->lchild); /* 递归构造左子树 */
if((*T)->lchild) /* 有左孩子 */
(*T)->LTag=Link; /* 给左标志赋值(指针) */
CreateBiThrTree(&(*T)->rchild); /* 递归构造右子树 */
if((*T)->rchild) /* 有右孩子 */
(*T)->RTag=Link; /* 给右标志赋值(指针) */
}
}
BiThrTree pre; /* 全局变量,始终指向刚刚访问过的结点 */
void InThreading(BiThrTree p)
{ /* 通过中序遍历进行中序线索化,线索化之后pre指向最后一个结点。算法6.7 */
if(p) /* 线索二叉树不空 */
{
InThreading(p->lchild); /* 递归左子树线索化 */
if(!p->lchild) /* 没有左孩子 */
{
p->LTag=Thread; /* 左标志为线索(前驱) */
p->lchild=pre; /* 左孩子指针指向前驱 */
}
if(!pre->rchild) /* 前驱没有右孩子 */
{
pre->RTag=Thread; /* 前驱的右标志为线索(后继) */
pre->rchild=p; /* 前驱右孩子指针指向其后继(当前结点p) */
}
pre=p; /* 保持pre指向p的前驱 */
InThreading(p->rchild); /* 递归右子树线索化 */
}
}
void InOrderThreading(BiThrTree *Thrt,BiThrTree T)
{ /* 中序遍历二叉树T,并将其中序线索化,Thrt指向头结点。算法6.6 */
*Thrt=(BiThrTree)malloc(sizeof(BiThrNode));
if(!*Thrt) /* 生成头结点不成功 */
exit(OVERFLOW);
(*Thrt)->LTag=Link; /* 建头结点,左标志为指针 */
(*Thrt)->RTag=Thread; /* 右标志为线索 */
(*Thrt)->rchild=*Thrt; /* 右指针回指 */
if(!T) /* 若二叉树空,则左指针回指 */
(*Thrt)->lchild=*Thrt;
else
{
(*Thrt)->lchild=T; /* 头结点的左指针指向根结点 */
pre=*Thrt; /* pre(前驱)的初值指向头结点 */
InThreading(T); /* 中序遍历进行中序线索化,pre指向中序遍历的最后一个结点 */
pre->rchild=*Thrt; /* 最后一个结点的右指针指向头结点 */
pre->RTag=Thread; /* 最后一个结点的右标志为线索 */
(*Thrt)->rchild=pre; /* 头结点的右指针指向中序遍历的最后一个结点 */
}
}
void InOrderTraverse_Thr(BiThrTree T,void(*Visit)(TElemType))
{ /* 中序遍历线索二叉树T(头结点)的非递归算法。算法6.5 */
BiThrTree p;
p=T->lchild; /* p指向根结点 */
while(p!=T)
{ /* 空树或遍历结束时,p==T */
while(p->LTag==Link) /* 由根结点一直找到二叉树的最左结点 */
p=p->lchild;
Visit(p->data); /* 访问此结点 */
while(p->RTag==Thread&&p->rchild!=T) /* p->rchild是线索(后继),且不是遍历的最后一个结点 */
{
p=p->rchild;
Visit(p->data); /* 访问后继结点 */
}
p=p->rchild; /* 若p->rchild不是线索(是右孩子),p指向右孩子,返回循环,*/
} /* 找这棵子树中序遍历的第1个结点 */
}
void PreThreading(BiThrTree p)
{ /* PreOrderThreading()调用的递归函数 */
if(!pre->rchild) /* p的前驱没有右孩子 */
{
pre->rchild=p; /* p前驱的后继指向p */
pre->RTag=Thread; /* pre的右孩子为线索 */
}
if(!p->lchild) /* p没有左孩子 */
{
p->LTag=Thread; /* p的左孩子为线索 */
p->lchild=pre; /* p的左孩子指向前驱 */
}
pre=p; /* 移动前驱 */
if(p->LTag==Link) /* p有左孩子 */
PreThreading(p->lchild); /* 对p的左孩子递归调用preThreading() */
if(p->RTag==Link) /* p有右孩子 */
PreThreading(p->rchild); /* 对p的右孩子递归调用preThreading() */
}
void PreOrderThreading(BiThrTree *Thrt,BiThrTree T)
{ /* 先序线索化二叉树T,头结点的右指针指向先序遍历的最后1个结点 */
*Thrt=(BiThrTree)malloc(sizeof(BiThrNode));
if(!*Thrt) /* 生成头结点 */
exit(OVERFLOW);
(*Thrt)->LTag=Link; /* 头结点的左指针为孩子 */
(*Thrt)->RTag=Thread; /* 头结点的右指针为线索 */
(*Thrt)->rchild=*Thrt; /* 头结点的右指针指向自身 */
if(!T) /* 空树 */
(*Thrt)->lchild=*Thrt; /* 头结点的左指针也指向自身 */
else
{ /* 非空树 */
(*Thrt)->lchild=T; /* 头结点的左指针指向根结点 */
pre=*Thrt; /* 前驱为头结点 */
PreThreading(T); /* 从头结点开始先序递归线索化 */
pre->rchild=*Thrt; /* 最后一个结点的后继指向头结点 */
pre->RTag=Thread;
(*Thrt)->rchild=pre; /* 头结点的后继指向最后一个结点 */
}
}
void PreOrderTraverse_Thr(BiThrTree T,void(*Visit)(TElemType))
{ /* 先序遍历线索二叉树T(头结点)的非递归算法 */
BiThrTree p=T->lchild; /* p指向根结点 */
while(p!=T) /* p没指向头结点(遍历的最后1个结点的后继指向头结点) */
{
Visit(p->data); /* 访问根结点 */
if(p->LTag==Link) /* p有左孩子 */
p=p->lchild; /* p指向左孩子(后继) */
else /* p无左孩子 */
p=p->rchild; /* p指向右孩子或后继 */
}
}
void PostThreading(BiThrTree p)
{ /* PostOrderThreading()调用的递归函数 */
if(p) /* p不空 */
{
PostThreading(p->lchild); /* 对p的左孩子递归调用PostThreading() */
PostThreading(p->rchild); /* 对p的右孩子递归调用PostThreading() */
if(!p->lchild) /* p没有左孩子 */
{
p->LTag=Thread; /* p的左孩子为线索 */
p->lchild=pre; /* p的左孩子指向前驱 */
}
if(!pre->rchild) /* p的前驱没有右孩子 */
{
pre->RTag=Thread; /* p前驱的右孩子为线索 */
pre->rchild=p; /* p前驱的后继指向p */
}
pre=p; /* 移动前驱 */
}
}
void PostOrderThreading(BiThrTree *Thrt,BiThrTree T)
{ /* 后序递归线索化二叉树 */
*Thrt=(BiThrTree)malloc(sizeof(BiThrNode));
if(!*Thrt) /* 生成头结点 */
exit(OVERFLOW);
(*Thrt)->LTag=Link; /* 头结点的左指针为孩子 */
(*Thrt)->RTag=Thread; /* 头结点的右指针为线索 */
if(!T) /* 空树 */
(*Thrt)->lchild=(*Thrt)->rchild=*Thrt; /* 头结点的左右指针指向自身 */
else
{ /* 非空树 */
(*Thrt)->lchild=(*Thrt)->rchild=T; /* 头结点的左右指针指向根结点(最后一个结点) */
pre=*Thrt; /* 前驱为头结点 */
PostThreading(T); /* 从头结点开始后序递归线索化 */
if(pre->RTag!=Link) /* 最后一个结点没有右孩子 */
{
pre->rchild=*Thrt; /* 最后一个结点的后继指向头结点 */
pre->RTag=Thread;
}
}
}
void DestroyBiTree(BiThrTree *T)
{ /* DestroyBiThrTree调用的递归函数,T指向根结点 */
if(*T) /* 非空树 */
{
if((*T)->LTag==0) /* 有左孩子 */
DestroyBiTree(&(*T)->lchild); /* 销毁左孩子子树 */
if((*T)->RTag==0) /* 有右孩子 */
DestroyBiTree(&(*T)->rchild); /* 销毁右孩子子树 */
free(*T); /* 释放根结点 */
T=NULL; /* 空指针赋0 */
}
}
void DestroyBiThrTree(BiThrTree *Thrt)
{ /* 初始条件:线索二叉树Thrt存在。操作结果:销毁线索二叉树Thrt */
if(*Thrt) /* 头结点存在 */
{
if((*Thrt)->lchild) /* 根结点存在 */
DestroyBiTree(&(*Thrt)->lchild); /* 递归销毁头结点lchild所指二叉树 */
free(*Thrt); /* 释放头结点 */
*Thrt=NULL; /* 线索二叉树Thrt指针赋0 */
}
}