题目:

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

分析:

根据二叉树的前序和中序遍历,重建二叉树。

我们知道:

前序遍历:根节点,左子树,右子树。

中序遍历:左子树,根节点,右子树。

后序遍历:左子树,右子树,根节点。

可以发现,前序遍历的第一个数便是整个数的根节点,而这个数在中序遍历中,又将数组分成两部分,其中左边便是左子树,右边是右子树,根据划分出来的左右子树,我们又可以返回前序遍历,去查看左右子树数组中的第一个元素,这样就可以递归求解这个问题了。

我们要记录好每次递归执行时的序列索引,这样有利于我们求解子问题需要数组的索引,前序遍历数组的左右索引记为leftPre,rightPre,中序遍历数组的左右索引是leftIn和rightIn。

则左子树的前序遍历数组左索引是leftPre+1,因为leftPre指的元素是root。

左子树的前序遍历数组右索引是leftPre+flag-leftIn,其中flag是root在中序遍历数组中的索引,而flag-leftIn正好是左子树元素的个数。

左子树的中序遍历数组索引是leftPre和flag-1,因为flag是我们找到的root元素,左边自然就是新的中序遍历数组了。

右子树的前序遍历数组左索引是leftPre+flag-leftIn+1,其实也就是左子树的右索引加1,因为在前序遍历中,左右子树是相连的。rightPre自然就成了右索引。

右子树的中序遍历数组索引是flag+1和rightIn。

不过在求解中,每次都要在中序遍历的数组中去查找根节点的索引,我们可以开始的时候将索引存进map中,需要的时候直接取,这样可以降低时间复杂度。

程序:

C++

class Solution {
public:
TreeNode* reConstructBinaryTree(vector<int> pre,vector<int> vin) {
return helper(pre, vin, , pre.size()-, , vin.size()-);
}
TreeNode* helper(vector<int>& preorder, vector<int>& inorder, int leftPre, int rightPre, int leftIn, int rightIn){
if(leftPre > rightPre)
return nullptr;
TreeNode* root = new TreeNode(preorder[leftPre]);
if(leftPre == rightPre)
return root;
else{
int i = leftIn;
for(;i <= rightIn; ++i){
if(inorder[i] == preorder[leftPre])
break;
}
root->left = helper(preorder, inorder, leftPre+, leftPre+i-leftIn, leftIn, i-);
root->right = helper(preorder, inorder, leftPre+i+-leftIn, rightPre, i+, rightIn);
return root;
}
}
};

Java

//use HashMap
import java.util.HashMap;
public class Solution {
public TreeNode reConstructBinaryTree(int [] pre,int [] in) {
map = new HashMap<>();
for(int i = 0; i < in.length; ++i){
map.put(in[i], i);
}
return helper(pre, in, 0, pre.length-1, 0, in.length-1);
}
public TreeNode helper(int[] preorder, int[] inorder, int leftPre, int rightPre, int leftIn, int rightIn){
if(leftPre > rightPre)
return null;
TreeNode root = new TreeNode(preorder[leftPre]);
if(leftPre == rightPre)
return root;
else{
int i = map.get(preorder[leftPre]);
root.left = helper(preorder, inorder, leftPre+1, leftPre+i-leftIn, leftIn, i-1);
root.right = helper(preorder, inorder, leftPre+i-leftIn+1, rightPre, i+1, rightIn);
return root;
}
}
private HashMap<Integer, Integer> map;
}

剑指Offer-4.重建二叉树(C++/Java)的更多相关文章

  1. 剑指Offer:重建二叉树【7】

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

  2. 《剑指offer》重建二叉树

    本题来自<剑指offer> 重构二叉树 题目: 输入某二叉树的前序遍历和中序遍历的结果,请重建出该二叉树.假设输入的前序遍历和中序遍历的结果中都不含重复的数字.例如输入前序遍历序列{1,2 ...

  3. 【Java】 剑指offer(6) 重建二叉树

    本文参考自<剑指offer>一书,代码采用Java语言.  更多:<剑指Offer>Java实现合集 题目 输入某二叉树的前序遍历和中序遍历的结果,请重建出该二叉树.假设输入的 ...

  4. Go语言实现:【剑指offer】重建二叉树

    该题目来源于牛客网<剑指offer>专题. 输入某二叉树的前序遍历和中序遍历的结果,请重建出该二叉树.假设输入的前序遍历和中序遍历的结果中都不含重复的数字.例如输入前序遍历序列{1,2,4 ...

  5. 剑指offer之重建二叉树

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

  6. 剑指OFFER之重建二叉树(九度OJ1385)

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

  7. 剑指offer:重建二叉树

    重建二叉树的前置知识: 0.遍历二叉树: (1)前序遍历:根左右 --> 先访问根节点,再前序遍历左子树,最后前序遍历右子树: (2)中序遍历:左根右 --> 先中序遍历左子树,再访问根节 ...

  8. 剑指Offer 4. 重建二叉树 (二叉树)

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

  9. 【剑指offer】重建二叉树

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

  10. 剑指offer——04重建二叉树(Python3)

    思路:在数据结构中,有一个条件反射,谈及二叉树,就递归.所以在实现重建二叉树时,也应该用到递归的思想. 在前序遍历中,根节点处于第一个:在中序遍历中,根节点的左边为左子树节点,根节点右边为右子树节点. ...

随机推荐

  1. Redux使用

    思想 应用中所有的state都以一个对象树的形式储存在一个单一的store中.唯一能改变state的办法是触发action,一个描述发生什么的对象.为了描述action如何改变state树,需要编写r ...

  2. Educational Codeforces Round 76 (Rated for Div. 2) E. The Contest dp

    E. The Contest A team of three programmers is going to play a contest. The contest consists of

  3. Educational Codeforces Round 76 (Rated for Div. 2) C. Dominated Subarray 水题

    C. Dominated Subarray Let's call an array

  4. flag 履行我的flag

    以后数组开小就不吃饭!!!!!! 上午考试不吃午饭 下午考试不吃晚饭 晚上考试不吃早饭 我以后还能吃饭吗 11.12距离csp-s还有2天,我的数组开小了,履行承诺,不吃饭了

  5. java之三元运算符

    逻辑运算 ? m : n;如果逻辑运算为真,则返回m,否则返回n 实例: 判断i,j两个数的大小,如果a较大,则输出1,否则输出0: 找到i,j,k三个数中的最大值: public class Tes ...

  6. 向github中已创建好的repository提交文件

    git init git remote add origin git@github.com:taishan1994/learn_django.git git pull origin master gi ...

  7. 《细说PHP》 第四版 样章 第二章 PHP的应用与发展 4

    2.4  PHP的发展 最初创建时,PHP是一个简单的用Perl语言编写的程序,只是为了统计自己的网站有多少访问者.后来又用C语言重新编写,多年来,PHP经过无数开源贡献者的不断迭代,历经数个版本,已 ...

  8. vue实现页面跳转(简易版)

    1.用点击函数 <button class="btntop" @click="gootherpage">跳转页面</button> 函数 ...

  9. 安装Goland开发工具

    安装Goland开发工具 开发工具: 文本类的编辑器:记事本,notepad,sublime text,atom... ​ 通过命令执行程序 IED:集成开发环境(integrated develop ...

  10. c# winform 窗体失去焦点关闭(钩子实现)

    先来一个辅助类 using System; using System.Collections.Generic; using System.Linq; using System.Runtime.Inte ...