剑指offer_面试题6_重建二叉树(分解步骤,逐个击破)
题目:输入某二叉树的前序遍历和中序遍历的结果。请重建出该二叉树。如果输入的前序遍历和中序遍历的结果中都不含反复的数字。
比如:输入前序遍历 {1,2,4,7,3,5,6,8} 和中序遍历序列 {4,7,2,1,5,3,8,6},则重建出图2.6所看到的的二叉树并输出它的头结点。
感触:复杂问题,要将它分解成一个个小问题,逐个击破,从而解决大问题。
我们都知道一个概念。知道 前序遍历 和 中序遍历。能够唯一确定一颗二叉树。
因此,我们首先须要怎样依据这两个已知条件,来画出这颗二叉树。在解决这道题的时候,最好用纸和笔。自己先实现一遍,然后分析自己实现的过程。分解每个步骤,而问题的算法就是依据你解决这个问题的步骤而来。
我在做的过程中,曾參考书中代码,然而别人的代码终究不是自己的。是别人的思路,自己在理解的过程中,总有这样那样的问题。因此。最好将书上的代码放一边,自己动手
去实践。依照自己的思路来解决整个问题。这样你在測试的时候。才有针对性。当然对于书中代码呈现的解决思路,须要选择性吸收。
对于这个问题,我把它分成三步:
1、首先拿到前序遍历序列,其作用就是用来取得 根节点。
(须要理解,二叉树是一个递归的过程,其子树亦是一个二叉树。每一次都是取得 二叉树的根节点)
2、找到 “前序遍历中取得的根节点” 在中序遍历序列中的位置
3、找到该位置后,就能够确定 该根节点 所接的左右子树(例如以下图所看到的),然后对这 子 二叉树进行递归操作。
这样又从第一步開始了。依次取得每一棵子二叉树的根节点。从而完毕整棵树的重建。
代码例如以下:
在解决这个问题的过程中,常常出现考虑问题不全面的问题,比方 以下代码中,加的一个函数 Judge_array()。它用来測试当你输入的前序序列和中序序列 不匹配的情况。这一点一開始乜有考虑到。
须要 铭记。警醒!
剑指offer中。有句话说的非常好,“最好在写代码前,考虑好測试用例“。在撸代码的过程中,特别有感触。
/****************************************************************************/
/** 题目:输入某二叉树的前序遍历和中序遍历的结果,请重建出该二叉树。 */
/** 如果输入的前序遍历和中序遍历的结果中都不含反复的数字。比如 */
/** 输入前序遍历{1,2,4,7,3,5,6,8}和中序遍历序列{4,7,2,1,5,3,8,6} */
/** 则重建出图2.6所看到的的二叉树并输出它的头结点。 */
/** 时间:2015.7.25 作者:jwt */
/****************************************************************************/
#include <iostream> using namespace std; typedef struct node {
int value;
struct node *lchild;
struct node *rchild;
}BiTree_Node; /**用于推断前序遍历序列和中序遍历数列书否匹配*/
bool Judge_array(int *pre,int *in,int n)
{
int i, j;
int k = 0;
for(i = 0; i < n; i++)
{
for(j = 0; j < n; j++)
{
if(pre[i] == in[j])
k++;
}
}
if(k != n)
return false;
else
return true;
} /*重建二叉树的递归操作,对于这个问题。用指针更加方便 */
BiTree_Node *Construct(int *pre_start, int *in_start, int len)
{
/**对于前序遍历数列,仅仅做一件事,取得根节点root */
int root_value = pre_start[0];
BiTree_Node *root;
root = new BiTree_Node;
root->value = root_value;
root->lchild = NULL;
root->rchild = NULL; /**在中序遍历中寻找上面取得的root的位置*/
int k = 0;
int new_len;
while(root_value != in_start[k] && k < len)
{
k++;
}
new_len = k; /**依据所找到的位置,推断左右子树*/
if(new_len > 0)
{
root->lchild = Construct(pre_start+1, in_start, new_len);
/*new_len 表示左子树的长度*/
}
if(len - new_len - 1> 0)
{
root->rchild = Construct(pre_start+new_len+1, in_start+new_len+1, len-new_len-1);
/*len-new_len-1 表示右子树的长度*/
} return root;
} /*重建二叉树函数*/
BiTree_Node *Construct_BiTree(int pre[],int in[],int length)
{
if(pre == NULL || in == NULL || length <= 0 || !Judge_array(pre, in, length))
return NULL;
else
return Construct(pre, in, length);
} /**二叉树后序遍历。通过后序遍历推断重建的二叉树是否正确*/
void Postorder(BiTree_Node *root)
{
if(NULL == root)
return;
else{
Postorder(root->lchild);
Postorder(root->rchild);
cout << root->value << ' ';
}
} int main()
{
int pre[8] = {1,2,4,7,3,5,6,8};
int in[8] = {4,7,2,1,5,3,8,6}; //int pre[5] = {1,2,3,4,5};
//int in[5] = {5,4,3,2,1}; //int pre[6] = {1,2,3,6,7,8};
//int in[6] = {3,2,1,6,7,8}; //int pre[5] = {1,2,3,4,5};
//int in[5] = {1,2,3,4,6}; BiTree_Node *root = Construct_BiTree(pre,in,8);
Postorder(root);
return 0;
}
结果例如以下:(能够用主函数凝视部分。測试其它情况)
/*点滴积累,我的一小步O(∩_∩)O~*/
剑指offer_面试题6_重建二叉树(分解步骤,逐个击破)的更多相关文章
- 剑指offer面试题6 重建二叉树(c)
- 剑指offer面试题6 重建二叉树(java)
注:(1)java中树的构建 (2)构建子树时可以直接利用Arrays.copyOfRange(preorder, from, to),这个方法是左开右闭的 package com.xsf.SordF ...
- 剑指Offer_编程题之重建二叉树
题目描述 输入某二叉树的前序遍历和中序遍历的结果,请重建出该二叉树.假设输入的前序遍历和中序遍历的结果中都不含重复的数字.例如输入前序遍历序列{1,2,4,7,3,5,6,8}和中序遍历序列{4,7, ...
- 剑指Offer:面试题6——重建二叉树(java实现)
问题描述:输入某二叉树的前序遍历和中序遍历的结果,请重建出该二叉树.假设输入的前序遍历和中序遍历的结果中都不包含重复的数字. 例如: 输入:前序{1,2,4,7,3,5,6,8},中序{4,7,2,1 ...
- C++版 - 剑指Offer 面试题39:二叉树的深度(高度)(二叉树深度优先遍历dfs的应用) 题解
剑指Offer 面试题39:二叉树的深度(高度) 题目:输入一棵二叉树的根结点,求该树的深度.从根结点到叶结点依次经过的结点(含根.叶结点)形成树的一条路径,最长路径的长度为树的深度.例如:输入二叉树 ...
- 剑指Offer - 九度1385 - 重建二叉树
剑指Offer - 九度1385 - 重建二叉树2013-11-23 23:53 题目描述: 输入某二叉树的前序遍历和中序遍历的结果,请重建出该二叉树.假设输入的前序遍历和中序遍历的结果中都不含重复的 ...
- 剑指offer【04】- 重建二叉树(java)
题目:重建二叉树 考点:树 题目描述:输入某二叉树的前序遍历和中序遍历的结果,请重建出该二叉树.假设输入的前序遍历和中序遍历的结果中都不含重复的数字.例如输入前序遍历序列{1,2,4,7,3,5,6, ...
- 剑指offer(4)重建二叉树
题目描述 输入某二叉树的前序遍历和中序遍历的结果,请重建出该二叉树.假设输入的前序遍历和中序遍历的结果中都不含重复的数字.例如输入前序遍历序列{1,2,4,7,3,5,6,8}和中序遍历序列{4,7, ...
- 剑指offer——面试题8:二叉树的下一个节点
// 面试题8:二叉树的下一个结点 // 题目:给定一棵二叉树和其中的一个结点,如何找出中序遍历顺序的下一个结点? // 树中的结点除了有两个分别指向左右子结点的指针以外,还有一个指向父结点的指针. ...
随机推荐
- Find or Query Data with C# Driver
https://docs.mongodb.com/getting-started/csharp/query/ Overview You can use the Find and FindAsync m ...
- 升级JDK9后eclipse无法启动的解决方法
解决方法-打开: D:\Program Files\eclipse\eclipse.ini 在文件末尾添加一行: --add-modules=ALL-SYSTEM 再次启动eclipse即可 感谢ht ...
- 关于Fragment的setUserVisibleHint() 方法和onCreateView()的执行顺序
1:setUserVisibleHint(boolean isVisibleToUser)的方法就很重要,根据方法名来看当前页面是否可见, 里面的boolean值就是判断当前页面是否可见的变量,所以大 ...
- JTable表格案例
package com.szht.gpy.frame; import java.awt.BorderLayout; import java.awt.event.ActionEvent; import ...
- caffe(11) 图像数据转换成db文件
在深度学习的实际应用中,我们经常用到的原始数据是图片文件,如jpg,jpeg,png,tif等格式的,而且有可能图片的大小还不一致.而在caffe中经常使用的数据类型是lmdb或leveldb,因此就 ...
- 洛谷P2839 [国家集训队]middle 主席树_二分
Code: #include <cstdio> #include <algorithm> #include <cstring> #include <strin ...
- linux下python3源码安装及卸载
Linux下Python3的源码编译安装和卸载方法 [日期:2019-06-21] 来源:博客园 作者:wuli潇萧 [字体:大 中 小] (一)Linux下软件的源码编译安装和卸载方法 L ...
- JDBC连接ORACLE无法登陆java.sql.SQLException: ORA-01017: invalid username/password; logon denied
当用jdbc连接Oracle数据库的时候 private Connection getConnection() throws SQLException { OracleDataSource ods = ...
- 紫书 例题 10-27 UVa 10214(欧拉函数)
只看一个象限简化问题,最后答案乘4+4 象限里面枚举x, 在当前这条固定的平行于y轴的直线中 分成长度为x的一段段.符合题目要求的点gcd(x,y) = 1 那么第一段1<= y <= x ...
- 洛谷 P1294 高手去散步
P1294 高手去散步 题目背景 高手最近谈恋爱了.不过是单相思.“即使是单相思,也是完整的爱情”,高手从未放弃对它的追求.今天,这个阳光明媚的早晨,太阳从西边缓缓升起.于是它找到高手,希望在晨读开始 ...