【算法】递归+深搜+哈希表:889.根据前序和后序遍历构造二叉树

目录

1、题目链接

相似题目:

2、题目

​3、解法(针对无重复值,哈希表+递归)

函数头-----找出重复子问题

函数体---解决子问题

4、代码


1、题目链接

889.根据前序和后序遍历构造二叉树(LeetCode)

相似题目:

105.从前序与中序遍历序列构造二叉树(LeetCode)

106.从中序与后序遍历序列构造二叉树(LeetCode)

2、题目

3、解法(针对无重复值,哈希表+递归)

前序:根节点、左子树、右子树
后序:左子树、右子树、根节点

二叉树的根节点一定是前序遍历的第一个节点,也是后序遍历的最后一个节点。

接下来,我们需要确定二叉树的左子树范围和右子树范围。

首先,仅凭前序和后序,得不到唯一的二叉树。结果可能有多个。

没有中序遍历的情况下,无法区分哪些节点属于左子树,哪些属于右子树。

第二个元素我们默认他是左子树的根节点。

函数头-----找出重复子问题

深搜函数头:preorder前序数组,

root :根节点在前序的下标

left:根节点所在子树的左边界

right:根节点所在子树的右边界

复制代码
TreeNode* dfs(const vector<int>& preorder, int root, int left, int right)

函数体---解决子问题

哈希表 hash,它存储了后序遍历中每个节点的位置。在函数的开头,我们可以找到任意节点在后序遍历中的位置。

  1. 如果 left >right ,说明范围为空,直接返回空节点。
  2. 否则,我们构造一个新的节点 node ,它的值为前序遍历中的第一个节点的值,也就是 preorder[root]。
  3. 如果 left 等于 right ,说明 root 没有左子树也没有右子树,直接返回 root。
  4. 否则,左子树的根节点的值为 preorder[ root + 1],我们在后序遍历中找到 preorder[ root + 1] 的位置,记为 pos_root。
    1. 那么左子树的节点个数 left_size = pos_root − left + 1,由此可知左子树后序遍历中的范围是 [left , pos_root ],右子树在后序遍历中的范围是 [pos_root + 1, right −1]。
  5. 知道了左右子树的范围,我们就可以递归地重建左右子树,然后将左右子树的根节点分别作为 node 的左右子节点。最后返回 node。
cpp 复制代码
        if (left > right)
            return NULL;

        TreeNode* node = new TreeNode(preorder[root]);//构建根节点
       
        //非常重要!!!影响整个代码的通过
        //root 没有左子树也没有右子树,直接返回 root。
        if (left == right)
            return node;

        int pos_root = hash[preorder[root + 1]];    //左子树的root在后序的下标
        
        int left_size = pos_root - left + 1; //左子树结点数(含root)
        int right_root = root + left_size + 1;//右子树根节点在前序中的下标

        node->left = dfs(preorder, root+1, left, pos_root);
        node->right = dfs(preorder, right_root, pos_root + 1, right - 1);

4、代码

cpp 复制代码
class Solution {

    //仅仅依靠前序、后序,会有存在多个二叉树
public:
    unordered_map<int, int> hash;

    //root 在pre的下标
    TreeNode* dfs(const vector<int>& preorder, int root, int left, int right)
    {
        if (left > right)
            return NULL;

        TreeNode* node = new TreeNode(preorder[root]);//构建根节点
       
        //非常重要!!!影响整个代码的通过
        //root 没有左子树也没有右子树,直接返回 root。
        if (left == right)
            return node;

        int pos_root = hash[preorder[root + 1]];    //左子树的root在后序的下标
        
        int left_size = pos_root - left + 1; //左子树结点数(含root)
        int right_root = root + left_size + 1;//右子树根节点在前序中的下标

        node->left = dfs(preorder, root+1, left, pos_root);
        node->right = dfs(preorder, right_root, pos_root + 1, right - 1);

        return node;
    }

    TreeNode* constructFromPrePost(vector<int>& preorder, vector<int>& postorder) {
        //postorder填充hash
        for (int i = 0; i < postorder.size(); i++)
        {
            hash[postorder[i]] = i;
        }

        return dfs(preorder, 0, 0, postorder.size() - 1);
    }
};
相关推荐
睡醒了叭6 分钟前
图像分割-传统算法-区域分割
图像处理·人工智能·算法·计算机视觉
代码游侠21 分钟前
复习——网络基础知识
网络·笔记·网络协议·算法·http
沈阳信息学奥赛培训31 分钟前
CCF GESP 2025/12/24 模拟测试 C++ 4级 编程题2
数据结构·算法
dddddppppp12332 分钟前
c 模拟一个fat16文件系统
c语言·c++·算法
行走的bug...1 小时前
利用计算机辅助数学运算
人工智能·算法·机器学习
CoderCodingNo1 小时前
【GESP】C++五级真题(数论-素数、贪心思想考点) luogu-B4050 [GESP202409 五级] 挑战怪物
开发语言·c++·算法
小O的算法实验室1 小时前
2026年AEI SCI1区TOP,基于多策略集成粒子群算法+无人机平滑覆盖路径规划,深度解析+性能实测
算法·论文复现·智能算法·智能算法改进
不一样的故事1262 小时前
高速采集箱
算法·信号处理
会员果汁2 小时前
算法-拓扑排序-C
c语言·开发语言·算法
YGGP2 小时前
【Golang】LeetCode 72. 编辑距离
算法·leetcode