0
  • 聊天消息
  • 系统消息
  • 评论与回复
登录后你可以
  • 下载海量资料
  • 学习在线课程
  • 观看技术视频
  • 写文章/发帖/加入社区
创作中心

完善资料让更多小伙伴认识你,还能领取20积分哦,立即完善>

3天内不再提示

Offer系列面试题0:重建二叉树

算法与数据结构 来源:图解面试算法 2020-07-09 15:03 次阅读

今天分享的题目来源于 LeetCode 上的剑指 Offer 系列面试题07. 重建二叉树,近半年在微软面试环节出现过 2 次,属于中高难度的算法题!

一、题目描述

输入某二叉树的前序遍历和中序遍历的结果,请重建该二叉树。假设输入的前序遍历和中序遍历的结果中都不含重复的数字。

例如,给出

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

返回如下的二叉树:

3 / 920 / 157

限制:

0 <= 节点个数 <= 5000

二、题目解析

首先,我们先来复习一下前序遍历、中序遍历。(在下方的视频中分布讲解)

前序遍历

二叉树的前序遍历顺序是:根节点、左子树、右子树,每个子树的遍历顺序同样满足前序遍历顺序。

中序遍历

二叉树的中序遍历顺序是:左子树、根节点、右子树,每个子树的遍历顺序同样满足中序遍历顺序。

复习过后,我们可以得出以下结论:

在二叉树的前序遍历序列中,第一个数字总是树的根结点的值;

在二叉树的中序遍历序列中,根结点的值在序列的中间,左子树的结点的值位于根结点的值的左边,而右子树的结点的值位于根结点的值的右边

以本题的序列为例,前序遍历序列的第一个数字 3 就是根结点的值,在中序遍历序列,找到根结点值的位置。根据中序遍历特点,在根结点的值 3前面的数字都是左子树结点的值,在根结点的值 3后面的数字都是右子树结点的值。

二叉树很重要的一个性质是递归,在找到了左子树、右子树的前序遍历序列和中序遍历序列后,我们可以按照同样的方法去确定子左子树和子右子树的构建。

具体的代码编写思路如下(来源于 Krahets's Blog):

递推参数:前序遍历中根节点的索引pre_root_idx、中序遍历左边界in_left_idx、中序遍历右边界in_right_idx。

终止条件:当in_left_idx > in_right_idx,子树中序遍历为空,说明已经越过叶子节点,此时返回 null 。

递推工作:

建立根节点 root :值为前序遍历中索引为pre_root_idx的节点值。

搜索根节点 root 在中序遍历的索引 i :为了提升搜索效率,本题解使用哈希表map预存储中序遍历的值与索引的映射关系,每次搜索的时间复杂度为 O(1)。

构建根节点root的左子树和右子树:通过调用 recursive()方法开启下一层递归。

左子树:根节点索引为 pre_root_idx + 1 ,中序遍历的左右边界分别为 in_left_idx 和 i - 1。

右子树:根节点索引为 i - in_left_idx + pre_root_idx + 1(即:根节点索引 + 左子树长度 + 1),中序遍历的左右边界分别为 i + 1 和 in_right_idx。

返回值:返回root,含义是当前递归层级建立的根节点root为上一递归层级的根节点的左或右子节点。

三、动画描述

四、图片描述

五、参考代码

classSolution{ //在中序序列中查找与前序序列首结点相同元素的时候,如果使用while循环去一个个找效率很慢 //这里我们借助数据结构HashMap来辅助查找,在开始递归之前把所有的中序序列的元素和它们所在的下标存到一个map中,这样查找的时间复杂度是O(logn) HashMapmap=newHashMap<>(); //保留的前序遍历 int[]preorder; publicTreeNodebuildTree(int[]preorder,int[]inorder){ this.preorder=preorder; //在开始递归之前把所有的中序序列的元素和它们所在的下标存到一个map中 for(inti=0;i< preorder.length; i++) {             map.put(inorder[i], i);         }         //二叉树的重要性质是递归         return recursive(0,0,inorder.length-1);     }     /** 根据前序遍历序列和中序遍历序列重新组建二叉树      * @param pre_root_idx 前序遍历的索引      * @param in_left_idx  中序遍历左边界的索引      * @param in_right_idx 中序遍历右边界的索引      */     public TreeNode recursive(int pre_root_idx, int in_left_idx, int in_right_idx) {         //子树中序遍历为空,说明已经越过叶子节点,此时返回 nul         if (in_left_idx >in_right_idx){ returnnull; } //root_idx是在前序里面的 TreeNoderoot=newTreeNode(preorder[pre_root_idx]); //通过map,根据前序的根节点的值,在中序中获取当前根的索引 intidx=map.get(preorder[pre_root_idx]); //左子树的根节点就是左子树的(前序遍历)第一个,就是+1,左边边界就是left,右边边界是中间区分的idx-1 root.left=recursive(pre_root_idx+1,in_left_idx,idx-1); //右子树的根,就是右子树(前序遍历)的第一个,就是当前根节点加上左子树的数量 root.right=recursive(pre_root_idx+(idx-1-in_left_idx+1)+1,idx+1,in_right_idx); returnroot; } }

这段代码的一个难点就是root.left与root.right,我这里抽离出来详细解释一下。

1、root.left

2、root.right

六、复杂度分析

时间复杂度

时间复杂度为 O(N)。

空间复杂度

空间复杂度为 O(N)。

七、相关标签

递归

哈希表

八、参考来源

1、https://leetcode-cn.com/problems/zhong-jian-er-cha-shu-lcof/solution/mian-shi-ti-07-zhong-jian-er-cha-shu-di-gui-fa-qin/ 题解区

2、https://krahets.gitee.io/views/sword-for-offer/2020-02-24-sword-for-offer-07.html

声明:本文内容及配图由入驻作者撰写或者入驻合作网站授权转载。文章观点仅代表作者本人,不代表电子发烧友网立场。文章及其配图仅供工程师学习之用,如有内容侵权或者其他违规问题,请联系本站处理。 举报投诉
  • 数字
    +关注

    关注

    1

    文章

    1689

    浏览量

    51074
  • 二叉树
    +关注

    关注

    0

    文章

    74

    浏览量

    12229

原文标题:面试字节跳动时,我竟然遇到了原题……

文章出处:【微信号:TheAlgorithm,微信公众号:算法与数据结构】欢迎添加关注!文章转载请注明出处。

收藏 人收藏

    评论

    相关推荐

    树与二叉树的定义

    树型结构 是一类重要的 非线性数据结构 ,其中以树和二叉树最为常用,直观来看,树是以分支关系定义的层次结构。树型结构在客观世界中广泛存在,比如人类社会中的祖辈关系,社会机构组织等等都可以用树来形象
    的头像 发表于 11-24 15:57 508次阅读
    树与<b class='flag-5'>二叉树</b>的定义

    硬件工程师经典面试题详解

    硬件工程师经典面试题详解
    的头像 发表于 11-20 15:08 715次阅读
    硬件工程师经典<b class='flag-5'>面试题</b>详解

    30道Linux面试题总结

    如果你是一名开发人员、系统管理员,或是仅仅对 Linux 感兴趣,那么这个列表是为你准备的。它包含了类 Unix 系统管理或编程职位面试中涉及 Linux 相关的所有常见问题。
    发表于 10-27 15:29 727次阅读
    30道Linux<b class='flag-5'>面试题</b>总结

    c语言面试题集(完整版)

    电子发烧友网站提供《c语言面试题集(完整版).pdf》资料免费下载
    发表于 10-20 11:20 1次下载
    c语言<b class='flag-5'>面试题</b>集(完整版)

    平衡二叉树(3)#数据结构

    数据函数二叉树
    未来加油dz
    发布于 :2023年09月05日 09:45:06

    平衡二叉树(2)#数据结构

    数据函数二叉树
    未来加油dz
    发布于 :2023年09月05日 09:43:54

    平衡二叉树(1)#数据结构

    数据函数二叉树
    未来加油dz
    发布于 :2023年09月05日 09:42:30

    二叉树二叉树的性质(3)#数据结构

    数据函数二叉树
    未来加油dz
    发布于 :2023年09月05日 09:09:51

    二叉树二叉树的性质(2)#数据结构

    数据函数二叉树
    未来加油dz
    发布于 :2023年09月05日 09:08:32

    二叉树二叉树的性质(1)#数据结构

    数据函数二叉树
    未来加油dz
    发布于 :2023年09月05日 09:06:44

    二叉排序树(2)(2)#数据结构

    数据函数二叉树
    未来加油dz
    发布于 :2023年09月05日 09:04:56

    这么简单的二叉树算法都不会?

    这个题目是leetcode的第572题,要求是这样的:给定两颗二叉树A和B,判断B是否是A的子树。
    的头像 发表于 08-29 11:19 496次阅读
    这么简单的<b class='flag-5'>二叉树</b>算法都不会?

    常见的嵌入式C语言面试题

    数组是最基本的数据结构,关于数组的面试题也屡见不鲜,本文罗列了一些常见的面试题,仅供参考。目前有以下18道题目。
    发表于 07-18 10:46 580次阅读

    【经典面试题】请使用C语言编程实现对IPV4地址的合法性判断

    【经典面试题】请使用C语言编程实现对IPV4地址的合法性判断
    的头像 发表于 05-16 15:23 1117次阅读

    分享10道有趣的嵌入式C语言面试题及答案

    10个C语言面试题,涉及指针、进程、运算、结构体、函数、内存,看看你能做出几个!
    的头像 发表于 05-09 10:54 1653次阅读