根据前序和中序遍历构建二叉树

一、题目

给定两个整数数组 preorderinorder ,其中 preorder 是二叉树的先序遍历inorder 是同一棵树的中序遍历,请构造二叉树并返回其根节点。

示例 1:

复制代码
输入: preorder = [3,9,20,15,7], inorder = [9,3,15,20,7]
输出: [3,9,20,null,null,15,7]

示例 2:

复制代码
输入: preorder = [-1], inorder = [-1]
输出: [-1]

提示:

  • 1 <= preorder.length <= 3000
  • inorder.length == preorder.length
  • -3000 <= preorder[i], inorder[i] <= 3000
  • preorderinorder无重复 元素
  • inorder 均出现在 preorder
  • preorder 保证 为二叉树的前序遍历序列
  • inorder 保证 为二叉树的中序遍历序列

二、源代码

objectivec 复制代码
/**
 * Definition for a binary tree node.
 * struct TreeNode {
 *     int val;
 *     struct TreeNode *left;
 *     struct TreeNode *right;
 * };
 */


struct TreeNode* buildTree(int* preorder, int preorderSize, int* inorder, int inorderSize){
    if(preorderSize == 0 || inorderSize == 0)
        return NULL;
    
    struct TreeNode* root = (struct TreeNode*)malloc(sizeof(struct TreeNode));
    int index;
    root -> val = preorder[0];
    for(index = 0;index < inorderSize;index++)
        if(inorder[index] == preorder[0])
            break;
    root -> left = buildTree(preorder + 1,index,inorder,index);
    root -> right = buildTree(preorder + index + 1,preorderSize - index - 1,inorder + index + 1,preorderSize - index - 1);
    return root;
}

三、解题思路

  1. 递归终止条件
objectivec 复制代码
if(preorderSize == 0 || inorderSize == 0)

    return NULL;

当子树的前序 / 中序数组长度为 0 时,说明这个子树没有节点,返回NULL(比如叶子节点的左 / 右子树)。

  1. 确定根节点
objectivec 复制代码
root -> val = preorder[0];

前序遍历的第一个元素必然是当前子树的根节点,这是核心依据。

  1. 找中序数组中的根节点下标
objectivec 复制代码
for(index = 0;index < inorderSize;index++)

    if(inorder[index] == preorder[0])

        break;

(1)中序数组中,根节点左侧的所有元素都是左子树的节点;

(2)右侧的所有元素都是右子树的节点;

(3)index的值就是左子树的节点个数。

  1. 递归构建左子树
objectivec 复制代码
root -> left = buildTree(preorder + 1, index, inorder, index);

(1)preorder + 1:前序数组跳过根节点,从第二个元素开始是左子树的前序遍历;

(2)index:左子树的节点个数(中序数组中根左侧的元素数);

(3)inorder:中序数组从开头到根节点前,都是左子树的中序遍历;

(4)最终返回的NULL或节点,作为当前根的左孩子。

  1. 递归构建右子树
objectivec 复制代码
root -> right = buildTree(preorder + index + 1, preorderSize - index - 1, inorder + index + 1, preorderSize - index - 1);

(1)preorder + index + 1:前序数组跳过根 + 左子树,剩下的是右子树的前序遍历;

(2)preorderSize - index - 1:右子树的节点个数(总节点数 - 根 - 左子树节点数);

(3)inorder + index + 1:中序数组跳过根节点,剩下的是右子树的中序遍历;

(4)最终返回的NULL或节点,作为当前根的右孩子。

四、总结

这段代码的核心是利用前序 / 中序的遍历特性,通过递归逐层拆解子树,最终还原整个二叉树

相关推荐
_OP_CHEN11 分钟前
【算法基础篇】(五十七)线性代数之矩阵乘法从入门到实战:手撕模板 + 真题详解
线性代数·算法·矩阵·蓝桥杯·c/c++·矩阵乘法·acm/icpc
天天爱吃肉821816 分钟前
【跨界封神|周杰伦×王传福(陶晶莹主持):音乐创作与新能源NVH测试,底层逻辑竟完全同源!(新人必看入行指南)】
python·嵌入式硬件·算法·汽车
im_AMBER16 分钟前
Leetcode 114 链表中的下一个更大节点 | 删除排序链表中的重复元素 II
算法·leetcode
xhbaitxl29 分钟前
算法学习day38-动态规划
学习·算法·动态规划
多恩Stone29 分钟前
【3D AICG 系列-6】OmniPart 训练流程梳理
人工智能·pytorch·算法·3d·aigc
历程里程碑31 分钟前
普通数组----轮转数组
java·数据结构·c++·算法·spring·leetcode·eclipse
pp起床32 分钟前
贪心算法 | part02
算法·leetcode·贪心算法
sin_hielo32 分钟前
leetcode 1653
数据结构·算法·leetcode
2501_9011478335 分钟前
面试必看:优势洗牌
笔记·学习·算法·面试·职场和发展
李日灐36 分钟前
C++进阶必备:红黑树从 0 到 1: 手撕底层,带你搞懂平衡二叉树的平衡逻辑与黑高检验
开发语言·数据结构·c++·后端·面试·红黑树·自平衡二叉搜索树