【LetMeFly】889.根据前序和后序遍历构造二叉树:分治(递归)——双O(n)的做法,五彩斑斓的题解(若不是彩色的可以点击原文链接查看)

本文主要是介绍【LetMeFly】889.根据前序和后序遍历构造二叉树:分治(递归)——双O(n)的做法,五彩斑斓的题解(若不是彩色的可以点击原文链接查看),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

【LetMeFly】889.根据前序和后序遍历构造二叉树:分治(递归)——双O(n)的做法,五彩斑斓的题解(若不是彩色的可以点击原文链接查看)

力扣题目链接:https://leetcode.cn/problems/construct-binary-tree-from-preorder-and-postorder-traversal/

给定两个整数数组,preorder 和 postorder ,其中 preorder 是一个具有 无重复 值的二叉树的前序遍历,postorder 是同一棵树的后序遍历,重构并返回二叉树。

如果存在多个答案,您可以返回其中 任何 一个。

 

示例 1:

输入:preorder = [1,2,4,5,3,6,7], postorder = [4,5,2,6,7,3,1]
输出:[1,2,3,4,5,6,7]

示例 2:

输入: preorder = [1], postorder = [1]
输出: [1]

 

提示:

  • 1 <= preorder.length <= 30
  • 1 <= preorder[i] <= preorder.length
  • preorder 中所有值都 不同
  • postorder.length == preorder.length
  • 1 <= postorder[i] <= postorder.length
  • postorder 中所有值都 不同
  • 保证 preorder 和 postorder 是同一棵二叉树的前序遍历和后序遍历

方法一:分治(递归)——双O(n)的做法

做这道题前强烈建议先去做一下从前序与中序建树。我们知道:

  • 前序遍历: 左子树 右子树
  • 后序遍历:左子树 右子树

我们需要明白:左子树和右子树只有一个的情况下,仅仅通过前序遍历和后续遍历的结果是无法得到原树是左子还是右子的。这是因为,对于某个只有一个子树的节点:

  • 假设此节点只有左子树,那么遍历结果为:前序【 左子树】后序【左子树
  • 假设此节点只有右子树,那么遍历结果为:前序【 右子树】后序【右子树

仅凭遍历结果无法得知到底是左子树还是右子树

因此我们可以按照“只有一个子树的情况下将其还原为左子树”的方式进行建树。

因此我们可以写一个函数dfs接收前序遍历数组后序遍历数组作为参数:

  1. 前序遍历数组的第一个节点(或者说后序遍历数组的最后一个节点)为

  2. 如果前序遍历数组的长度为1,那么说明只有节点,直接返回。

  3. 否则必定存在左子树(前面我们得出了结论,即使只有一个子树也可以视为左子树)。因此我们只需要得到左子树右子树(可能为空但无所谓)的长度,就能在前序遍历数组后序遍历数组中将二者划分出来,并继续递归。确定左子树长度的方法为:

    前序遍历数组中,左子树的第一个节点为左子树的根节点。

    后序遍历数组中,左子树的最后一个节点为左子树的根节点。

    因此从前序遍历数组中可以得到左子树的根节点,由这个节点在后序遍历数组中的位置,能得到左子树的长度。

    从而右子树的长度也能从任意一个遍历数组中,减去左子树的长度(减节点的长度)得出。

递归的终止条件为“前序遍历数组为空”,此时返回空节点。

Tips: 可以在预处理时建立一个哈希表,以便能快速地找到左子树的根节点在后序遍历数组中的位置。

  • 时间复杂度 O ( N ) O(N) O(N),其中 N N N是节点个数
  • 空间复杂度 O ( N ) O(N) O(N)

AC代码

C++
class Solution {
private:unordered_map<int, vector<int>::iterator> ma;TreeNode* dfs(vector<int>::iterator preLeft, vector<int>::iterator preRight, vector<int>::iterator postLeft, vector<int>::iterator postRight) {if (preLeft >= preRight) {return nullptr;}if (preLeft + 1 == preRight) {  // 只有一个节点return new TreeNode(*preLeft);}int leftLength = leftLength = ma[*(preLeft + 1)] - postLeft + 1;  // 注意是*(preLeft + 1)return new TreeNode(*preLeft,dfs(preLeft + 1, preLeft + 1 + leftLength, postLeft, postLeft + leftLength),dfs(preLeft + 1 + leftLength, preRight, postLeft + leftLength, postRight - 1));}
public:TreeNode* constructFromPrePost(vector<int>& preOrder, vector<int>& postOrder) {for (vector<int>::iterator it = postOrder.begin(); it != postOrder.end(); it++) {ma[*it] = it;}return dfs(preOrder.begin(), preOrder.end(), postOrder.begin(), postOrder.end());}
};
Python
# from typing import List, Optional# # Definition for a binary tree node.
# class TreeNode:
#     def __init__(self, val=0, left=None, right=None):
#         self.val = val
#         self.left = left
#         self.right = rightclass Solution:def dfs(self, preOrder: List[int], preLeft: int, preRight: int, postOrder: List[int], postLeft: int, postRight: int) -> Optional[TreeNode]:if preLeft >= preRight:return Noneif preLeft + 1 == preRight:return TreeNode(preOrder[preLeft])leftLength = self.ma[preOrder[preLeft + 1]] - postLeft + 1return TreeNode(preOrder[preLeft],self.dfs(preOrder, preLeft + 1, preLeft + 1 + leftLength, postOrder, postLeft, postLeft + leftLength),self.dfs(preOrder, preLeft + 1 + leftLength, preRight, postOrder, postLeft + leftLength, postRight - 1))def constructFromPrePost(self, preOrder: List[int], postOrder: List[int]) -> TreeNode:self.ma = dict()for i in range(len(postOrder)):self.ma[postOrder[i]] = ireturn self.dfs(preOrder, 0, len(preOrder), postOrder, 0, len(postOrder))

同步发文于CSDN,原创不易,转载经作者同意后请附上原文链接哦~
Tisfy:https://letmefly.blog.csdn.net/article/details/136227823

这篇关于【LetMeFly】889.根据前序和后序遍历构造二叉树:分治(递归)——双O(n)的做法,五彩斑斓的题解(若不是彩色的可以点击原文链接查看)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/735785

相关文章

C语言函数递归实际应用举例详解

《C语言函数递归实际应用举例详解》程序调用自身的编程技巧称为递归,递归做为一种算法在程序设计语言中广泛应用,:本文主要介绍C语言函数递归实际应用举例的相关资料,文中通过代码介绍的非常详细,需要的朋... 目录前言一、递归的概念与思想二、递归的限制条件 三、递归的实际应用举例(一)求 n 的阶乘(二)顺序打印

mysql如何查看当前连接数

《mysql如何查看当前连接数》:本文主要介绍mysql如何查看当前连接数问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录mysql查看当前连接数查看mysql数据库允许最大连接数总结mysql查看当前连接数查看当前连接数SHOW STATUS LIKE

查看Oracle数据库中UNDO表空间的使用情况(最新推荐)

《查看Oracle数据库中UNDO表空间的使用情况(最新推荐)》Oracle数据库中查看UNDO表空间使用情况的4种方法:DBA_TABLESPACES和DBA_DATA_FILES提供基本信息,V$... 目录1. 通过 DBjavascriptA_TABLESPACES 和 DBA_DATA_FILES

Jackson库进行JSON 序列化时遇到了无限递归(Infinite Recursion)的问题及解决方案

《Jackson库进行JSON序列化时遇到了无限递归(InfiniteRecursion)的问题及解决方案》使用Jackson库进行JSON序列化时遇到了无限递归(InfiniteRecursi... 目录解决方案‌1. 使用 @jsonIgnore 忽略一个方向的引用2. 使用 @JsonManagedR

Python如何查看数据的类型

《Python如何查看数据的类型》:本文主要介绍Python如何查看数据的类型方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录python查看数据的类型1. 使用 type()2. 使用 isinstance()3. 检查对象的 __class__ 属性4.

Windows命令之tasklist命令用法详解(Windows查看进程)

《Windows命令之tasklist命令用法详解(Windows查看进程)》tasklist命令显示本地计算机或远程计算机上当前正在运行的进程列表,命令结合筛选器一起使用,可以按照我们的需求进行过滤... 目录命令帮助1、基本使用2、执行原理2.1、tasklist命令无法使用3、筛选器3.1、根据PID

Rust中的BoxT之堆上的数据与递归类型详解

《Rust中的BoxT之堆上的数据与递归类型详解》本文介绍了Rust中的BoxT类型,包括其在堆与栈之间的内存分配,性能优势,以及如何利用BoxT来实现递归类型和处理大小未知类型,通过BoxT,Rus... 目录1. Box<T> 的基础知识1.1 堆与栈的分工1.2 性能优势2.1 递归类型的问题2.2

使用Python快速实现链接转word文档

《使用Python快速实现链接转word文档》这篇文章主要为大家详细介绍了如何使用Python快速实现链接转word文档功能,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 演示代码展示from newspaper import Articlefrom docx import

C++中使用vector存储并遍历数据的基本步骤

《C++中使用vector存储并遍历数据的基本步骤》C++标准模板库(STL)提供了多种容器类型,包括顺序容器、关联容器、无序关联容器和容器适配器,每种容器都有其特定的用途和特性,:本文主要介绍C... 目录(1)容器及简要描述‌php顺序容器‌‌关联容器‌‌无序关联容器‌(基于哈希表):‌容器适配器‌:(

Python如何使用seleniumwire接管Chrome查看控制台中参数

《Python如何使用seleniumwire接管Chrome查看控制台中参数》文章介绍了如何使用Python的seleniumwire库来接管Chrome浏览器,并通过控制台查看接口参数,本文给大家... 1、cmd打开控制台,启动谷歌并制定端口号,找不到文件的加环境变量chrome.exe --rem