题目链接地址:

pid=1385">http://ac.jobdu.com/problem.php?pid=1385

题目1385:重建二叉树

时间限制:1 秒内存限制:32 兆特殊判题:否提交:4441解决:1321

题目描写叙述:

输入某二叉树的前序遍历和中序遍历的结果,请重建出该二叉树。假设输入的前序遍历和中序遍历的结果中都不含反复的数字。比如输入前序遍历序列{1,2,4,7,3,5,6,8}和中序遍历序列{4,7,2,1,5,3,8,6},则重建二叉树并输出它的后序遍历序列。

输入:

输入可能包括多个測试例子,对于每一个測试案例,

输入的第一行为一个整数n(1<=n<=1000):代表二叉树的节点个数。

输入的第二行包括n个整数(当中每一个元素a的范围为(1<=a<=1000)):代表二叉树的前序遍历序列。

输入的第三行包括n个整数(当中每一个元素a的范围为(1<=a<=1000)):代表二叉树的中序遍历序列。

输出:

相应每一个測试案例,输出一行:

假设题目中所给的前序和中序遍历序列能构成一棵二叉树。则输出n个整数。代表二叉树的后序遍历序列,每一个元素后面都有空格。

假设题目中所给的前序和中序遍历序列不能构成一棵二叉树,则输出”No”。

例子输入:

8

1 2 4 7 3 5 6 8

4 7 2 1 5 3 8 6

8

1 2 4 7 3 5 6 8

4 1 2 7 5 3 8 6

例子输出:

7 4 2 5 8 6 3 1

No


分析:

採用递归重构二叉树

(左代表左子树。右代表右子树,根代表根结点)

    前序遍历:根-左-右

    中序遍历:左-根-右

    后序遍历:左-右-根

定理:给定某棵二叉树的中序遍历序列和前序遍历序列(或者后序遍历序列)就能唯一构造出该二叉树。



原因——由于通过前序遍历(或者后序遍历)能够找到二叉树的根结点。再依据根结点在中序遍历序列中的位置就能够确定根结点的左右子树。由于二叉树是一种递归结构,二叉树的左右子树也都是二叉树,所以递归依据前序和中序遍历序列能够确定各个结点之间的父子关系。

採用递归的思路将问题转化为本质同样可是规模更小的子问题。

前序遍历的第一个节点为根节点,依据根节点的值在中序遍历中找到其相应位置。左边是左子树,右边是右子树。然后左右递归求解就可以。

须要注意的是:假设中序遍历序列不包括前序遍历序列第一个元素则表明无法重构二叉树。

时间复杂度:

  每次在中序遍历中找根节点的位置须要O(n)的查找时间,推导复杂度:

    T(n) = 2 * T(n / 2) + O(1) + O(n)

    T(n) = O(n * log(n))

空间复杂度   

  递归求解,由于每一个节点都会被递归到。所以空间复杂度为O(n)。

 


代码:

/*********************************
-----------------------------------
【剑指Offer面试题】 九度OJ1385:重建二叉树
-----------------------------------
Author:牧之丶 Date:2015年
Email:bzhou84@163.com
**********************************/
#include<stdio.h>
#include <iostream>
using namespace std;
#define MAX 1005 // 二叉树的结点
typedef struct Node
{
int data; // 数据域
Node * lChild; // 左子树
Node * rChild; // 右子树
}BTNode;
BTNode bTNode[MAX]; bool RebuildBinaryTree; // 推断是否能重构二叉树
int preOrder[MAX];
int inOrder[MAX]; // 初始化二叉树中的每一个结点
void initBinaryTree(int n)
{
int i;
RebuildBinaryTree = true;
for(i = 0;i < n;i++)
{
bTNode[i].data = preOrder[i];
bTNode[i].lChild = NULL;
bTNode[i].rChild = NULL;
}
} //重构二叉树
void reBuildBinaryTree(int beginPreOrder,int endPreOrder,int beginInOrder,int endInOrder)
{
int i;
int position = -1; // 前序遍历序列第一个结点在中序遍历序列中的位置
bool flag = false; // 推断前序遍历序列中的第一个结点是否在中序遍历序列中
for(i = beginInOrder;i <= endInOrder;i++) // 遍历二叉树的中序遍历序列,得到根结点在中序遍历序列中的位置
{
if(preOrder[beginPreOrder] == inOrder[i])
{
position = i - beginInOrder;
flag = true;
break;
}
}
if(false == flag)
{
RebuildBinaryTree = false;
return;
}
else
{
//重构左子树
if(beginPreOrder + 1 <= beginPreOrder + position && beginInOrder <= beginInOrder + position - 1)
{
bTNode[beginPreOrder].lChild = &bTNode[beginPreOrder + 1];
reBuildBinaryTree(beginPreOrder + 1,beginPreOrder + position,beginInOrder,beginInOrder + position - 1);
}
//重构右子树
if(beginPreOrder + position + 1 <= endPreOrder && beginInOrder + position + 1 <= endInOrder)
{
bTNode[beginPreOrder].rChild = &bTNode[beginPreOrder + position + 1];
reBuildBinaryTree(beginPreOrder + position + 1,endPreOrder,beginInOrder + position + 1,endInOrder);
}
}
} //后序遍历输出
void postOrder(BTNode * root)
{
if(NULL == root)
return;
else
{
postOrder(root -> lChild);
postOrder(root -> rChild);
cout<<root -> data<<" ";
}
} int main()
{
int n;
while(cin>>n)
{
for(int i = 0;i < n;i++)
{
scanf("%d",&preOrder[i]);
}
for(int i = 0;i < n;i++)
{
scanf("%d",&inOrder[i]);
}
initBinaryTree(n);
reBuildBinaryTree(0,n - 1,0,n - 1);
if(false == RebuildBinaryTree)
cout<<"No"<<endl;
else
{
postOrder(&bTNode[0]); //bTNode[0]是根结点
cout<<endl;
}
}
return 0;
} /**************************************************************
Problem: 1385
Language: C++
Result: Accepted
Time:10 ms
Memory:1552 kb
****************************************************************/

【剑指Offer面试题】 九度OJ1385:重建二叉树的更多相关文章

  1. 《剑指offer》— JavaScript(4)重建二叉树

    重建二叉树 题目描述 输入某二叉树的前序遍历和中序遍历的结果,请重建出该二叉树.假设输入的前序遍历和中序遍历的结果中都不含重复的数字.例如输入前序遍历序列{1,2,4,7,3,5,6,8}和中序遍历序 ...

  2. 【剑指Offer面试题】 九度OJ1389:变态跳楼梯

    转自:http://www.myexception.cn/program/1973966.html 时间限制:1 秒内存限制:32 兆特殊判题:否提交:2331解决:1332 题目描述: 一只青蛙一次 ...

  3. 【剑指Offer面试题】 九度OJ1518:反转链表

    与其非常快写出一段漏洞百出的代码,倒不如细致分析再写出鲁棒的代码. 提前想好測试用例(输入非空等等)进行測试改动代码. 题目链接地址: http://ac.jobdu.com/problem.php? ...

  4. 【剑指Offer面试题】 九度OJ1368:二叉树中和为某一值的路径

    题目链接地址: http://ac.jobdu.com/problem.php? pid=1368 题目1368:二叉树中和为某一值的路径 时间限制:1 秒内存限制:32 兆特殊判题:否提交:2252 ...

  5. 【剑指Offer面试题】 九度OJ1517:链表中倒数第k个结点

    鲁棒性是指程序可以推断输入是否符合规范要求,并对不和要求的输入予以 合理的处理. 题目链接地址: http://ac.jobdu.com/problem.php?pid=1517 题目1517:链表中 ...

  6. 【剑指Offer面试题】 九度OJ1371:最小的K个数

    题目链接地址: http://ac.jobdu.com/problem.php?pid=1371 题目1371:最小的K个数 时间限制:1 秒内存限制:32 兆特殊判题:否提交:5938解决:1265 ...

  7. 【剑指Offer面试题】 九度OJ1516:调整数组顺序使奇数位于偶数前面

    题目链接地址: http://ac.jobdu.com/problem.php?pid=1516 题目1516:调整数组顺序使奇数位于偶数前面 时间限制:1 秒内存限制:128 兆特殊判题:否提交:2 ...

  8. 【剑指Offer面试题】九度OJ1384:二维数组中的查找

    下决心AC全部剑指offer面试题. 九度OJ面试题地址:http://ac.jobdu.com/hhtproblems.php 书籍:何海涛--<剑指Offer:名企面试官精讲典型编程题> ...

  9. 【剑指Offer面试题】 九度OJ1510:替换空格

    c/c++ 中的字符串以"\0"作为结尾符.这样每一个字符串都有一个额外字符的开销. 以下代码将造成内存越界. char str[10]; strcpy(str, "01 ...

随机推荐

  1. 添加QScintilla时显示无法解析的外部函数

    转载请注明出处:http://www.cnblogs.com/dachen408/p/7147165.html 问题:添加QScintilla时显示无法解析的外部函数 解决方案:去掉头文件qscisc ...

  2. Apache与IIS端口冲突解决方法

    在安装Apache或者php集成环境包是经常会遇到Apache的80端口被占用导致无法正常启动Apache. Win7可以通过如下方法解决(如果坚持要使用80端口的话): 1.打开"控制面板 ...

  3. crontab 每月最后一天执行命令

    没有什么是解决不了的事情,如果有,只是我们的知识不够精通,学得不扎实 需求:有一个程序,需要在每个月的最后一天执行 例如:每个月的最后一天早上8:00 打印 dede 到  /tmp/test.txt ...

  4. c Xcode has incompatible definitions in different translations units

    解决方案: build > setting enable module 改为No

  5. my @unpacking_list = values %map_function; print "\n".@unpacking_list; 输出是3 把 @unpacking_list 当做一个数 输出了

      my %map_function = (     88     "OK_func" => "open_statement",     89     & ...

  6. left_v2.js

    $(document).ready(function(){ $(".mc_left a").each(function(){ var href = $(this).attr(&qu ...

  7. zabbix源码安装后,设置为服务启动和关闭

    zabbix源码安装,使用service启动与关闭服务 1. zabbix客户端的系统服务脚本 1.1 拷贝启动脚本 zabbix的源码提供了系统服务脚本,在/usr/local/src/zabbix ...

  8. Division

    Description   Write a program that finds and displays all pairs of 5-digit numbers that between them ...

  9. ECNU 3260 袋鼠妈妈找孩子(dfs)

    链接:http://acm.ecnu.edu.cn/problem/3260/ 题意: 给出一个x,y,k.求从左上角到(x,y)最短路径不少于k而且最快到达(x,y)的迷宫.(迷宫有多个 输出其中一 ...

  10. //……关于TCP三次握手与四次挥手

    TCP的概述 TCP把连接作为最基本的对象,每一条TCP连接都有两个端点,这种断点我们叫作套接字(socket),它的定义为端口号拼接到IP地址即构成了套接字,例如,若IP地址为192.3.4.16 ...