二叉树重建[通俗易懂]

二叉树重建

大家好,又见面了,我是全栈君。

一、已知先序遍历和中序遍历。求后序遍历。

依据先序遍历和中序遍历还原二叉树的主要思想:

1、先序遍历序列的第一个元素必然是根节点,能够由此获取二叉树的根节点。

2、依据根节点,在中序遍历序列中查找该节点。由中序遍历的性质可知。中序遍历中该根节点左边的序列必然在根节点的左子树中,而根节点右边的序列必然在右子树中。由此能够知道先序遍历中左子树以及右子树的起止位置。

3、分别对左子树和右子树反复上述的过程,直至全部的子树的起止位置相等时。说明已经到达叶子节点,遍历完成。

实现代码:

#include<cstdio>#include<cstring>#include<cstdlib>struct Node{    char value;    Node *left, *right;};Node *build_new_node(char ch){    Node *p = (Node*)malloc(sizeof(Node));    p->value = ch;    p->left = p->right = NULL;    return p;}Node *Rebuild(char *pre, char *in, int n){    if(n == 0) return NULL;    char ch = pre[0];    Node *p = build_new_node(ch);    int i;    for(i = 0; i < n && in[i] != ch; i++);    int l_len = i;    int r_len = n - i - 1;    if(l_len > 0) p->left = Rebuild(pre+1, in, l_len);    if(r_len > 0) p->right = Rebuild(pre+l_len+1, in+l_len+1, r_len);    return p;}void PostOrder(Node *p){    if(p == NULL) return ;    PostOrder(p->left);    PostOrder(p->right);    printf("%c",p->value);}int main(){    char PreOrder[100], inOrder[100];    while(~scanf("%s%s",PreOrder, inOrder))    {        Node *root = Rebuild(PreOrder,inOrder,strlen(PreOrder));        PostOrder(root);        printf("\n");    }    return 0;}

二、已知中序遍历和后序遍历,求先序遍历。

http://acm.nyist.net/JudgeOnline/problem.php?pid=756

依据中序遍历和后序遍历还原二叉树的主要思想:

1、后序遍历序列的最后一个元素必然是根节点,能够由此获取二叉树的根节点。

2、依据根节点,在中序遍历序列中查找该节点,由中序遍历的性质可知,中序遍历中该根节点左边的序列必然在根节点的左子树中,而根节点右边的序列必然在右子树中。

由此能够知道后序遍历中左子树以及右子树的起止位置。

3、分别对左子树和右子树反复上述的过程,直至全部的子树的起止位置相等时,说明已经到达叶子节点。遍历完成。

实现代码:

#include<cstdio>#include<cstring>#include<cstdlib>struct Node{    char value;    Node *left, *right;};Node *build_new_node(char ch){    Node *p = (Node*)malloc(sizeof(Node));    p->value = ch;    p->left = p->right = NULL;    return p;}Node *Rebuild(char *post, char *in, int n){    if(n == 0) return NULL;    char ch = post[n-1];    Node *p = build_new_node(ch);    int i;    for(i = 0; i < n && in[i] != ch; i++);    int l_len = i;    int r_len = n - i - 1;    if(l_len > 0) p->left = Rebuild(post, in, l_len);    if(r_len > 0) p->right = Rebuild(post + l_len, in+l_len+1, r_len);    return p;}void PreOrder(Node *p){    if(p == NULL) return ;    printf("%c",p->value);    PreOrder(p->left);    PreOrder(p->right);}int main(){    char PostOrder[100], InOrder[100];    while(~scanf("%s%s",PostOrder, InOrder))    {        Node *root = Rebuild(PostOrder,InOrder,strlen(PostOrder));        PreOrder(root);        printf("\n");    }    return 0;}

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 举报,一经查实,本站将立刻删除。

发布者:全栈程序员-用户IM,转载请注明出处:https://javaforall.cn/116630.html原文链接:https://javaforall.cn

【正版授权,激活自己账号】: Jetbrains全家桶Ide使用,1年售后保障,每天仅需1毛

【官方授权 正版激活】: 官方授权 正版激活 支持Jetbrains家族下所有IDE 使用个人JB账号...

(0)


相关推荐

  • 中标麒麟和centos区别_中标麒麟debian

    中标麒麟和centos区别_中标麒麟debian首先参考网上常见的CentOS如何本地yum安装软件的:(后面是中标麒麟)1、首先进行光盘的挂载,注意光盘挂载时不会自动建立目录的,所以需要自己建立目录mkdir/mnt/cdrommount/dev/cdrom/mnt/cdrom#dev目录为设备目录2、更改本地源地址cd/etc/yum.repos.d/#可以看见CentOS-Base.repo和Cen…

  • Java内存管理-程序运行过程(一)「建议收藏」

    勿在流沙住高台,出来混迟早要还的。做一个积极的人编码、改bug、提升自己我有一个乐园,面向编程,春暖花开!相信在做Java开发的伙伴一定知道 JVM(Java Virtual Machine(Java虚拟机)!本系列会开启对JVM相关的知识的探索和总结,让我们一起踏入JVM的学习之旅中,去了解和发现更加精彩的Java世界! 正如奥古斯特·罗丹 说过:世界上不是缺少美,而是缺少发现美的眼…

  • ★ Android ExpandableListView中子元素无法点击 解决方案!

    ★ Android ExpandableListView中子元素无法点击 解决方案!

  • jQuery 模板 tmpl 用法「建议收藏」

    jQuery 模板 tmpl 用法「建议收藏」昨晚无意中发现一个有趣的jQuery插件.tmpl(),其文档在这里。官方解释对该插件的说明:将匹配的第一个元素作为模板,render指定的数据,签名如下:.tmpl([data,][options])其中参数data的用途很明显:用于render的数据,可以是任意js类型,包括数组和对象。options一般情况下都是

  • 关于STM32使用LAN8720A插拔网线重连「建议收藏」

    关于STM32使用LAN8720A插拔网线重连「建议收藏」关于STM32使用LAN8720A插拔网线重连其实在做这个功能的时候大家一定要心平气和,不要认为有多复杂,多看DATASHEET,当然后面会遇到一些问题,所以在踩过坑之后,过了差不多一年了,也算是回过头来做个记录吧。1.关于LAN8720的手册解读通过查阅lan8720的数据收册:标黄部分,在寄存器映射中第一个寄存器为基本状态寄存器,然后我们通过查阅这个寄存器发现,在该寄存器的bit2中说明了,当检测网线插入的时候该位为1,否则为0。知道这个那就好办了,我们可以根据这个状态位去判断网线的接入状

  • docker-Dockerfile文件详解

    docker-Dockerfile文件详解

发表回复

您的电子邮箱地址不会被公开。

关注全栈程序员社区公众号