剑指 Offer 07. 重建二叉树

前置概念:

前序:访问根节点,先序遍历左子树,先序遍历右子树;

中序:中序遍历左子树,访问根节点,中序遍历右子树;

后序:后序遍历左子树,后序遍历右子树,访问根节点;

图示

前序:A- B-D-G-H-E-C-F

中序:G-D-H-B-E-A-C-F

后序:G-H-D-E-B-F-C-A

怎么构建一棵树:

通过一组(前序-中序 或 中序-后序)来确定唯一一棵树;

以该题的测试样例举例:

前序遍历 preorder = [3,9,20,15,7]
中序遍历 inorder = [9,3,15,20,7]

前序的第一个数是根节点,在pre数组中找到根节点(3),然后在中序遍历中找到3的位置,那么中序遍历中3左边的(9)就是根节点的左子树(左节点),(15、20、7)就是根节点的右子树(右节点).

通过以上规律可得到一颗树,即:

解题思路如下:

重建二叉树的重构函数分为三个步骤

  • 首先要找到当前元素节点,即pre中的第一个,
  • 接着在中序遍历中找到它的左右子树(此处表现为对前序数组、中序数组的分割操作),以便构造他的左右孩子。
  • 最后再将左右子树在分别放到重构函数中

比如此时,我们找到3的右子树(20、15、7),如下图,它实质上适合上图一样的,只不过此时是以3为根节点,其他操作是一样。

前序:3-20-15-7

中序:3-15-20-7

  此时重复操作:找到当前元素节点,即前序中的第一个,接着便可以在中序遍历中找到他的左右子树,以便构造他的左右孩子。发现他没有左子树,将左孩子放入构造函数中,如图所示:

前序:20-15-7

中序:15-20-7

此时重复操作:找到当前元素节点,即前序中的第一个,接着便可以在中序遍历中找到他的左右子树,以便构造他的左右孩子。将左右孩子放入重构函数中,即

  直到分割成叶子节点,不存在左右子树,他无法再进行分割,故返回自己。叶子节点返回后,其父节点的左右子树分别有了指向,便返回,一步一步向上返回,最后会返回整个二叉树

通过两段代码来学习该题目:

主要是通过第一段代码来理解第二部分的思想。

第一部分代码段:

leetcode官方题解:详细解释就不说了,重点关注下:size_left_subtree变量(左子树中的节点数目);

  1. 通过map的key值可以找到中序中的根节点——————>构造哈希映射,帮助我们快速定位根节点
class Solution {
private Map<Integer, Integer> indexMap; public TreeNode myBuildTree(int[] preorder, int[] inorder, int preorder_left, int preorder_right, int inorder_left, int inorder_right) {
if (preorder_left > preorder_right) {
return null;
} // 前序遍历中的第一个节点就是根节点
int preorder_root = preorder_left;
// 在中序遍历中定位根节点
int inorder_root = indexMap.get(preorder[preorder_root]); // 先把根节点建立出来
TreeNode root = new TreeNode(preorder[preorder_root]);
// 得到左子树中的节点数目
int size_left_subtree = inorder_root - inorder_left;
// 递归地构造左子树,并连接到根节点
// 先序遍历中「从 左边界+1 开始的 size_left_subtree」个元素就对应了中序遍历中「从 左边界 开始到 根节点定位-1」的元素
root.left = myBuildTree(preorder, inorder, preorder_left + 1, preorder_left + size_left_subtree, inorder_left, inorder_root - 1);
// 递归地构造右子树,并连接到根节点
// 先序遍历中「从 左边界+1+左子树节点数目 开始到 右边界」的元素就对应了中序遍历中「从 根节点定位+1 到 右边界」的元素
root.right = myBuildTree(preorder, inorder, preorder_left + size_left_subtree + 1, preorder_right, inorder_root + 1, inorder_right);
return root;
} public TreeNode buildTree(int[] preorder, int[] inorder) {
int n = preorder.length;
// 构造哈希映射,帮助我们快速定位根节点
indexMap = new HashMap<Integer, Integer>();
for (int i = 0; i < n; i++) {
indexMap.put(inorder[i], i);
}
return myBuildTree(preorder, inorder, 0, n - 1, 0, n - 1);
}
}

第二部分代码段:

class Solution {
public TreeNode buildTree(int [] pre,int [] in) {
TreeNode root=ConstructCore(pre,0,pre.length-1,in,0,in.length-1);
return root;
} public TreeNode ConstructCore(int[] pre,int startPre,int endPre,int[] in,int startIn,int endIn)
{
//前序起点下标>前序终点下标 || 中序的起点下标 > 中序终点下标
if(startPre>endPre||startIn>endIn)
return null;
//通过起点下标找到前序中的根节点,并创建
TreeNode node = new TreeNode(pre[startPre]);
//中序遍历
for(int i=startIn;i<=endIn;i++)
{ //在中序中找到了根节点 此时i就是根节点的下标
if(in[i]==pre[startPre])
{
//ConstructCore(前序遍历,前序左子树起点下标,前序左子树终点下标,中序遍历,中序左子树起点下标,中序左子树终点下标)
node.left = ConstructCore(pre,startPre+1,startPre+i-startIn,in,startIn,i-1);
node.right =ConstructCore(pre,startPre+i-startIn+1,endPre,in,i+1,endIn);
break;
}
}
return node; }
}

上述代码最难理解部分如下:

for(int i=startIn;i<=endIn;i++)
{ //在中序中找到了根节点 此时i就是根节点的下标
if(in[i]==pre[startPre])
{
node.left = ConstructCore(pre,startPre+1,startPre+i-startIn,in,startIn,i-1);
node.right =ConstructCore(pre,startPre+i-startIn+1,endPre,in,i+1,endIn);
break;
}
}

我们分析node.left以及node.right:

node.left = ConstructCore(pre,startPre+1,startPre+i-startIn,in,startIn,i-1);

ConstructCore(前序遍历数组,前序左子树起点下标(除去根节点),前序左子树终点下标,中序遍历,中序左子树起点下标,中序左子树终点下标)

详解: startPre+i-startIn

“ i ” 是当前根节点的下标, i-startin --->表示在左子树中的节点数目,前序左子树

startPre+i-startIn--->表示了前序数组中左子树的终止下标

node.right =ConstructCore(pre,startPre+i-startIn+1,endPre,in,i+1,endIn);

ConstructCore(前序遍历,前序右子树起点下标,前序右子树终点下标,中序遍历,中序右子树起点下标,中序右子树终点下标)

i-startin --->表示在左子树中的节点数目,

startPre+i-startIn--->前序左子树终点下标

startPre+i-startIn+1 ---->前序右子树起点下标

i-1 -->中序右子树起点下标。

参考文献:

代码引自:重建二叉树--->https://www.cnblogs.com/MrSaver/p/9205436.html

LeetCode剑指offer官方题解

剑指Offer07 重建二叉树的更多相关文章

  1. 剑指Offer-4.重建二叉树(C++/Java)

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

  2. 牛客_剑指offer_重建二叉树,再后续遍历_递归思想_分两端

       总结:    重建二叉树:其实就是根据前序和中序重建得到二叉树,得到后续,只要输出那边设置输出顺序即可 [编程题]重建二叉树 输入某二叉树的前序遍历和中序遍历的结果,请重建出该二叉树.假设输入的 ...

  3. 剑指offer--4.重建二叉树

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

  4. 剑指Offer——重建二叉树

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

  5. 剑指Offer——重建二叉树2

    Question 输入某二叉树的后序遍历和中序遍历的结果,请重建出该二叉树.假设输入的后序遍历和中序遍历的结果中都不含重复的数字.例如输入后序遍历序列{1, 3, 4, 2}和中序遍历序列{1, 2, ...

  6. 剑指offer--19.重建二叉树

    先序:根>左>右 中序:左>根>右 后序:左>右>根 e.g. {1,2,4,7,3,5,6,8} {4,7,2,1,5,3,8,6} 先序第一个元素是根节点,在中 ...

  7. 用js刷剑指offer(重建二叉树)

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

  8. 剑指Offer04 重建二叉树

    代码有问题 /************************************************************************* > File Name: 04_ ...

  9. 剑指offer 重建二叉树

    1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 3 ...

  10. 《剑指offer》 二叉树的镜像

    本题来自<剑指offer>二叉树的镜像 题目: 操作给定的二叉树,将其变换为源二叉树的镜像. 二叉树的镜像定义:源二叉树 8 / \ 6 10 / \ / \ 5 7 9 11 镜像二叉树 ...

随机推荐

  1. COM组件开发-关于在开发环境下COM组件的(来自 HRESULT 的异常:0x80080005 (CO_E_SERVER_EXEC_FAILURE)) 以及 在CLR语言下可能报错 未能加载文件或程序集“Interop.xxx 的问题

    1.关于在开发环境下COM组件的(来自 HRESULT 的异常:0x80080005 (CO_E_SERVER_EXEC_FAILURE)) 开发环境下,COM组件注册的文件 不一定是你自己现在程序调 ...

  2. freeswitch号码变化方案

    概述 freeswitch是一款简单易用的开源音视频软交换平台. 在生产环境中,由于各个线路的号码规则并不统一,经常需要针对中继线路做号码变换的方案. 本文主要介绍fs中有哪些可选的号码变换方案. 环 ...

  3. python之configparser类的使用

    一.定义配置文件格式如下:data.conf [interface] url=http://192.168.37.8:7777/api/mytest2 [switch] switch_car=on [ ...

  4. Https 原理与工作流程及证书链校验

    本文为博主原创,未经允许不得转载: 目录 HTTP传输三大风险 安全通信原则 HTTPS定义 TLS/SSL 协议及加密算法 HTTPS工作流程 HTTPS协议和HTTP协议的区别 CA机构 证书链校 ...

  5. zookeeper源码(02)源码编译启动及idea导入

    本文介绍一下zookeeper-3.9.0源码下载.编译及本地启动. 下载源码 git clone https://gitee.com/apache/zookeeper.git cd zookeepe ...

  6. Advanced .Net Debugging 1:你必须知道的调试工具

    一.简介    我曾看到过许多开发人员使用错误的工具来分析问题,更有甚者,有些人连任何工具都没有使用.他们采取的分析方法通常包括:输出更多的调试信息,或者做一些临时性的代码审查.这里的临时性是指,通过 ...

  7. [转帖]Sql Server之旅——第六站 使用winHex利器加深理解数据页

    https://www.cnblogs.com/huangxincheng/p/4251770.html 这篇我来介绍一个winhex利器,这个工具网上有介绍,用途大着呢,可以用来玩数据修复,恢复删除 ...

  8. [转帖]clickHouse单机模式安装部署(RPM安装)

    关于版本和系统的选择 操作系统:Centos-7 ClickHouse: rpm 在安装,20.x 安装前的准备 CentOS7 打开文件数限 在 /etc/security/limits.conf ...

  9. [转帖]Nginx reuseport 导致偶发性卡顿

    https://github.com/jonmeredith/tcpperf https://plantegg.github.io/2023/06/08/Nginx%20reuseport%20%E5 ...

  10. 【转帖】JAVA GC日志分析

    https://zhuanlan.zhihu.com/p/613592552 ​ 目录 1. GC分类 针对HotSpot VM的实现,它里面的GC按照回收区域又分为两大种类型:一种是部分收集(Par ...