Java编程练习之:重建二叉树(牛客网题目)

1. 题目

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

2. 相关知识

  • 使用 Arrays. 方法的时候,要引用 import java.util.Arrays
  • TreeNode 是一个方法类,每建立一个新的节点,都会有一个 left 指针和一个 right指针,而且在使用方法类的时候,要用 new 来创建
  • Arrays.copyOfRange():copyOfRange(T[] original, int from, int to)
    • 原序列 original
    • 从哪个 index 开始
    • 到哪个 index 结束
    • 这是一个左闭右开的区间
  • 使用 while 的时候不要忘记:
    • i++
    • 完成任务后要 break

3. 解题思路

3.1 整体思路

根据 中序遍历前序遍历 可以确定二叉树,具体过程为:

  • 根据前序序列第一个结点确定根结点
  • 根据根结点在中序序列中的位置分割出左右两个子序列,也就是代码中的in_left_treein_right_tree,这两个序列就是下一级的二叉树的中序序列。
  • 同样的,在前序序列pre中,原来的序列,也被分成了两个新的序列pre_left_treepre_right_tree 作为下一级的二叉树的前序序列。
  • 对下一级的二叉树依然分别递归使用同样的方法继续分解

3.2 实例思路

前序序列{1,2,4,7,3,5,6,8} = pre
中序序列{4,7,2,1,5,3,8,6} = in

  • 根据当前前序序列的第一个结点确定根结点,为 1
  • 找到 1 在中序遍历序列中的位置,为 in[3]
  • 按照 in[3]位置切割左右子树,则 in[3] 前面的为左子树, in[3] 后面的为右子树,所以切割后的左子树前序序列为:{2,4,7},切割后的左子树中序序列为:{4,7,2};切割后的右子树前序序列为:{3,5,6,8},切割后的右子树中序序列为:{5,3,8,6}
    对子树分别使用同样的方法分解

4. 代码

/**
 * Definition for binary tree
 * public class TreeNode {
 *     int val;
 *     TreeNode left;
 *     TreeNode right;
 *     TreeNode(int x) { val = x; }
 * }
 */
 
import java.util.Arrays;

public class Solution {

    public TreeNode reConstructBinaryTree(int [] pre,int [] in) {
        
        if (pre.length == 0 || in.length == 0) {
            return null;
        }
        
        TreeNode root = new TreeNode(pre[0]);
       
        int i = 0;
        int[]pre_left_tree;
        int[]pre_right_tree;
        int[]in_left_tree;
        int[]in_right_tree;
        
        while(i<in.length){
            if(pre[0] == in[i]){
                pre_left_tree = Arrays.copyOfRange(pre,1,i+1);
                pre_right_tree = Arrays.copyOfRange(pre,i+1,pre.length);
                in_left_tree = Arrays.copyOfRange(in,0,i);
                in_right_tree = Arrays.copyOfRange(in,i+1,in.length);
                root.left = reConstructBinaryTree(pre_left_tree,in_left_tree);
                root.right = reConstructBinaryTree(pre_right_tree,in_right_tree);
                break;
            }
            i++;
            
        }
        return root;
    }

}
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

暖仔会飞

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值