IT数码 购物 网址 头条 软件 日历 阅读 图书馆
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
图片批量下载器
↓批量下载图片,美女图库↓
图片自动播放器
↓图片自动播放器↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁
 
   -> 数据结构与算法 -> 数据结构---线索二叉树 -> 正文阅读

[数据结构与算法]数据结构---线索二叉树

//
//   线索二叉树
//

# include<math.h>
# include<iostream>
# include<stdio.h>
using namespace std;

int num[30] = {1,2,4,0,0,5,0,0,3,6,0,0,7,0,0};
int i = 0;


typedef int Elemtype;

typedef struct ThreadTNode{     //链式存储结构
    Elemtype data;
    struct ThreadTNode *lchild,*rchild;
    int ltag = 0,rtag = 0;
}ThreadTNode,*ThreadTree;

ThreadTNode *pre = NULL;


void printNode(ThreadTNode *p){
    printf("%d   ",p->data);
}


//找中序前驱节点
ThreadTNode *find_LastInNode(ThreadTNode *p){    //找到以p为跟的子树中最后一个被中序遍历的点
    while (p->rtag==0)       //最后先被访问的节点没有右孩子,所以ltag=1
        p = p->rchild;        //这个不是正常的中序遍历,而是直接从右孩子那里去找
    return p;
}

ThreadTNode *find_PreInNode(ThreadTNode *p){    //找到p的前驱
    if(p->ltag == 0)    //如果p没有被线索化(即p有左孩子)
        return find_LastInNode(p->lchild);     //前驱就是左子树最后一个被访问的节点
    else if(p->ltag == 1)     //如果p被线索化,前驱就是左孩子
        return p->lchild;
}

void RevInOrder(ThreadTree T){    //这样就可以逆中序遍历二叉树
    for(ThreadTNode *p = find_LastInNode(T);p!=NULL;p=find_PreInNode(p)){
        printNode(p);
    }
}




//找中序后继节点
ThreadTNode *find_FirstInThread(ThreadTNode *p){
    while(p->ltag == 0)
        p = p->lchild;
    return p;
}

ThreadTNode *find_NextInNode(ThreadTNode *p){   //找p的后继节点
    if(p->rtag==0)
        return find_FirstInThread(p->rchild);     //如果p没有被线索化,那就是右子树第一个被访问的节点
    else if(p->rtag==1)
        return p->rchild;
}

void InOder(ThreadTree T){   //非递归中序遍历
    for(ThreadTNode *p = find_FirstInThread(T);p!=NULL;p = find_NextInNode(p))
        printNode(p);
}


//先序遍历找前驱困难,后序遍历找后继节点困难



void inThread(ThreadTree &p){      //一边遍历一边线索化
    if(p!=NULL){
        inThread(p->lchild);    //中间这一段相当于visit()

        if(p->lchild==NULL){     //左孩子为空,建立前驱线索
            p->lchild = pre;
            p->ltag = 1;
        }
        if(pre!=NULL && pre->rchild==NULL){    //建立这个节点的前驱节点没有右孩子(该改方向了)(没有直接后继)
            pre->rchild = p;
            pre->rtag = 1;
        }
        pre = p;

        printf("%d  ",p->data);

        inThread(p->rchild);
    }
}

void preThread(ThreadTree &p){      //一边遍历一边线索化
    if(p!=NULL){
            //中间这一段相当于visit()

        if(p->lchild==NULL){     //左孩子为空,建立前驱线索
            p->lchild = pre;
            p->ltag = 1;
        }
        if(pre!=NULL && pre->rchild==NULL){    //建立这个节点的前驱节点没有右孩子(该改方向了)
            pre->rchild = p;
            pre->rtag = 1;
        }
        pre = p;

        if(p->ltag==0)               //防止转圈
            preThread(p->lchild);

        preThread(p->rchild);
    }
}

void postThread(ThreadTree &p){      //一边遍历一边线索化
    if(p!=NULL){
        postThread(p->lchild);

        postThread(p->rchild);    //后序遍历不会出现这样的问题

        if(p->lchild==NULL){     //左孩子为空,建立前驱线索
            p->lchild = pre;
            p->ltag = 1;
        }
        if(pre!=NULL && pre->rchild==NULL){    //建立这个节点的前驱节点没有右孩子(该改方向了)
            pre->rchild = p;
            pre->rtag = 1;
        }
        pre = p;


    }
}



void visit(ThreadTree &T){
    if(T->lchild==NULL){
        T->lchild = pre;
        T->ltag = 1;
    }

    if(pre!=NULL && pre->rchild==NULL){
        pre->rchild = T;
        pre->rtag = 1;
    }

    pre = T;

    printf("%d  ",T->data);
}

void preOrder(ThreadTree &T){
    if(T!=NULL){
        visit(T);
        if(T->ltag==0)        //如果左孩子已经被线索化,那么它指向的就是前驱,遍历的时候又回去了
            preOrder(T->lchild);
        preOrder(T->rchild);
    }
}

void inOrder(ThreadTree &T){
    if(T!=NULL){
        inOrder(T->lchild);
        visit(T);
        inOrder(T->rchild);
    }
}

void postOrder(ThreadTree &T){
    if(T!=NULL){
        postOrder(T->lchild);
        postOrder(T->rchild);
        visit(T);
    }
}

void buildTree(ThreadTree &T){
    int el;
    el = num[i];i++;
//    scanf("%d",&el);
    if(el==0)
        T=NULL;
    else{
        T = (ThreadTNode *)malloc(sizeof(ThreadTNode));
        T->data = el;
        T->ltag = 0;
        T->rtag = 0;
        buildTree(T->lchild);
        buildTree(T->rchild);
    }

}

void createinThread1(ThreadTree &T){     //一边遍历一边线索化
    pre = NULL;

    if(T!=NULL){
        inOrder(T);
//        inThread(T);
        if(pre->rchild == NULL)     //这里也不用判断,最后一个节点的右孩子肯定是NULL
            pre->rtag = 1;          //处理遍历的最后一个节点
    }
}

void createinThread2(ThreadTree &T){     //一边遍历一边线索化
    pre = NULL;

    if(T!=NULL){
//        inOrder(T);
        inThread(T);
        if(pre->rchild == NULL)     //这里也不用判断,最后一个节点的右孩子肯定是NULL
            pre->rtag = 1;          //处理遍历的最后一个节点
    }
}



int main(void){
    ThreadTree T1;
    T1 = NULL;

    buildTree(T1);
//    createinThread1(T1);     //方法一
    createinThread2(T1);    //方法二
    printf("\n\n\n");
    RevInOrder(T1);
    printf("\n\n\n");
    InOder(T1);


    return 0;
}





最后用逆序和非递归可以进行验证.

  数据结构与算法 最新文章
【力扣106】 从中序与后续遍历序列构造二叉
leetcode 322 零钱兑换
哈希的应用:海量数据处理
动态规划|最短Hamilton路径
华为机试_HJ41 称砝码【中等】【menset】【
【C与数据结构】——寒假提高每日练习Day1
基础算法——堆排序
2023王道数据结构线性表--单链表课后习题部
LeetCode 之 反转链表的一部分
【题解】lintcode必刷50题<有效的括号序列
上一篇文章      下一篇文章      查看所有文章
加:2021-07-29 11:53:53  更:2021-07-29 11:56:49 
 
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁

360图书馆 购物 三丰科技 阅读网 日历 万年历 2025年1日历 -2025/1/28 11:50:55-

图片自动播放器
↓图片自动播放器↓
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
图片批量下载器
↓批量下载图片,美女图库↓
  网站联系: qq:121756557 email:121756557@qq.com  IT数码