树——004重建二叉树

1,问题描述

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

2,解题思路

解题思路(一):

(1)前序遍历序列中的第一元素是二叉树的根节点,题目中根节点的值是1。

(2)从中序遍历序列中找到根节点值1的下标(记为index=3)。那么中序遍历序列中index的左边就是左子树的中序遍历序列in[0,index](包左不包右),即[4,7,2];右边为右子树的中序遍历序列in[index+1, in.length](包左不包右),即[5,3,8,6]

(3)根据(2)中的下标值从前序遍历序列中找到根节点的左子树前序遍历序列pre[1,index+1],即[2,4,7]。那么右子树的前序遍历序列为pre[index+1,pre.length],即[3,5,6,8]

至此找到了该根节点左子树的前序遍历序列pre[1,index+1]、中序遍历序列in[0,index],以及右子树的前序遍历序列pre[index+1,pre.length]、中序遍历序列in[index+1, in.length]

注:java的Arrays.copyOfRange()方法,是包左不包右的。

(4)将左右子树看做是待构建的二叉树,再递归处理(1)(2)(3)步骤,递归出口是pre或者in任何一个为空就返回null;

解题思路(二):(牛客网大牛的思路,但是觉得不如上面的思路好理解哇,我是费了吃奶的劲才想明白的。。。我是菜鸟,请忽略)

(1)首先找到中序的根节点对应的i,然后i的左边就是左子树(startIn,i-1),右边就是右子树(i+1,endIn)

(2)然后前序的左子树的长度就可以用中序的左子树的长度(i-startIn)得到,那么左子树就是(startPre+1,startPre+1+(i-startIn-1)),右子树的起点就是左子树的终点+1即(startPre+1+(i-startIn-1)+1,endPre)。

(3)递归前两步,递归出口是,startPre>endPre || starIn>endIn,返回null。

3,源码

  • 解题思路一:
import java.util.Arrays;
/**
 * Definition for binary tree
 * public class TreeNode {
 *     int val;
 *     TreeNode left;
 *     TreeNode right;
 *     TreeNode(int x) { val = x; }
 * }
 */
public class Solution {
    public TreeNode reConstructBinaryTree(int [] pre,int [] in) {
       if(pre.length == 0||in.length == 0) return null;
        TreeNode head = new TreeNode(pre[0]);
        for(int i = 0; i < in.length; i++){
            if(pre[0] == in[i]){
                head.left = reConstructBinaryTree(Arrays.copyOfRange(pre, 1, i+1), 
                                                  Arrays.copyOfRange(in, 0, i));
                head.right = reConstructBinaryTree(Arrays.copyOfRange(pre, i+1, pre.length),
                                                   Arrays.copyOfRange(in, i+1,in.length));
                break;
            }
        }
        return head;
    }
}
  • 解题思路二:
/**
 * Definition for binary tree
 * public class TreeNode {
 *     int val;
 *     TreeNode left;
 *     TreeNode right;
 *     TreeNode(int x) { val = x; }
 * }
 */
public class Solution {
    public TreeNode reConstructBinaryTree(int [] pre,int [] in) {
        TreeNode root=reConstructBinaryTree(pre,0,pre.length-1,in,0,in.length-1);
        return root;
    }
    //前序遍历{1,2,4,7,3,5,6,8}和中序遍历序列{4,7,2,1,5,3,8,6}
    private TreeNode reConstructBinaryTree(int [] pre,int startPre,int endPre,int [] in,int startIn,int endIn) {
         
        if(startPre>endPre||startIn>endIn)
            return null;
        TreeNode root=new TreeNode(pre[startPre]);
         
        for(int i=startIn;i<=endIn;i++)
            if(in[i]==pre[startPre]){
                root.left=reConstructBinaryTree(pre,startPre+1,startPre+i-startIn,in,startIn,i-1);
                root.right=reConstructBinaryTree(pre,i-startIn+startPre+1,endPre,in,i+1,endIn);
                      break;
            }
                 
        return root;
    }
}

 

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值