本文参考自《剑指offer》一书,代码采用Java语言。

 更多:《剑指Offer》Java实现合集

题目

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

思路

  前序遍历第一个值就是根结点的值,根据该值在中序遍历的位置,可以轻松找出该根结点左右子树的前序遍历和中序遍历,之后又可以用同样方法构建左右子树,所以该题可以采用递归的方法完成。

  刚开始思考的时候,想的是构建一个遍历函数,输入为前序和中序遍历的数组,输出为根结点。但是这样的话每次都需要构建子树的数组,非常麻烦。

  之后想到,该函数的输入不一定要用数组,因为最初的前序和中序遍历数组已经有了,就直接用该数组的下标来表示子树的数组即可。

  即构建函数construct(int[] pre, int[] in, int pStart, int pEnd, int iStart, int iEnd),pre和in始终用最初前序遍历和中序遍历的数组代入,pStart、pEnd代表当前树的前序数组开始和结束位置,iStart、iEnd代表中序数组开始和结束位置。

测试用例

  1.正常二叉树

  2.左斜树

  3.右斜树

  4.单个结点

  5.数组为空

  6.前序与中序不匹配

完整Java代码

(含测试代码)

/**
*
* @Description 重建二叉树
*
* @author yongh
* @date 2018年9月12日 下午4:35:19
*/ // 题目:输入某二叉树的前序遍历和中序遍历的结果,请重建出该二叉树。假设输
// 入的前序遍历和中序遍历的结果中都不含重复的数字。例如输入前序遍历序列{1,
// 2, 4, 7, 3, 5, 6, 8}和中序遍历序列{4, 7, 2, 1, 5, 3, 8, 6},则重建出
// 二叉树并输出它的头结点。
public class ConstructBinaryTree {
class TreeNode {
int val; public TreeNode(int val) {
this.val = val;
} TreeNode left;
TreeNode right;
} public TreeNode reConstructBinaryTree(int[] pre, int[] in) {
if (pre == null || in == null || pre.length <= 0 || in.length <= 0 || pre.length != in.length) {
throw new RuntimeException("数组不符合规范!");
}
return construct(pre, in, 0, pre.length - 1, 0, in.length - 1);
} /**
*
* @Description 由前序遍历序列和中序遍历序列得到根结点
* pre、in:始终用最初的前序遍历和中序遍历数组代入
* pStart、pEnd:当前树的前序数组开始和结束位置
* iStart、iEnd:中序数组开始和结束位置
*/
public TreeNode construct(int[] pre, int[] in, int pStart, int pEnd, int iStart, int iEnd) {
TreeNode root = new TreeNode(pre[pStart]);
if (pStart == pEnd && iStart == iEnd) {
if (pre[pStart] != in[iStart])
throw new RuntimeException("数组不符合规范!");
return root;
}
int index = iStart; // 用于记录中序遍历序列中根结点的位置
while (root.val != in[index] && index <= iEnd) {
index++;
}
if (index > iEnd)
throw new RuntimeException("数组不符合规范!");
int leftLength = index - iStart;
if (leftLength > 0) {
root.left = construct(pre, in, pStart + 1, pStart + leftLength, iStart, index - 1);
}
if (leftLength < iEnd - iStart) {
root.right = construct(pre, in, pStart + leftLength + 1, pEnd, index + 1, iEnd);
}
return root;
} private void preOrderTraverse(TreeNode node) {
if (node == null)
return;
System.out.print(node.val);
preOrderTraverse(node.left);
preOrderTraverse(node.right);
} private void inOrderTraverse(TreeNode node) {
if (node == null)
return;
inOrderTraverse(node.left);
System.out.print(node.val);
inOrderTraverse(node.right);
} /**
* 正常二叉树
*/
public void test1() {
int[] pre = { 1, 2, 4, 7, 3, 5, 6, 8 };
int[] in = { 4, 7, 2, 1, 5, 3, 8, 6 };
TreeNode root = reConstructBinaryTree(pre, in);
System.out.print("test1:");
preOrderTraverse(root);
System.out.print("//");
inOrderTraverse(root);
System.out.println();
} /**
* 左斜树
*/
public void test2() {
int[] pre = { 1, 2, 3, 4, 5 };
int[] in = { 5, 4, 3, 2, 1 };
TreeNode root = reConstructBinaryTree(pre, in);
System.out.print("test2:");
preOrderTraverse(root);
System.out.print("//");
inOrderTraverse(root);
System.out.println();
} /**
* 右斜树
*/
public void test3() {
int[] pre = { 1, 2, 3, 4, 5 };
int[] in = { 1, 2, 3, 4, 5 };
TreeNode root = reConstructBinaryTree(pre, in);
System.out.print("test3:");
preOrderTraverse(root);
System.out.print("//");
inOrderTraverse(root);
System.out.println();
} /**
* 单个结点
*/
public void test4() {
int[] pre = { 1 };
int[] in = { 1 };
TreeNode root = reConstructBinaryTree(pre, in);
System.out.print("test4:");
preOrderTraverse(root);
System.out.print("//");
inOrderTraverse(root);
System.out.println();
} /**
* 数组为空
*/
public void test5() {
int[] pre = {};
int[] in = {};
TreeNode root = reConstructBinaryTree(pre, in);
System.out.print("test5:");
preOrderTraverse(root);
System.out.print("//");
inOrderTraverse(root);
System.out.println();
} public static void main(String[] args) {
ConstructBinaryTree demo = new ConstructBinaryTree();
demo.test1();
demo.test2();
demo.test3();
demo.test4();
demo.test5();
}
}

  

test1://
test2://
test3://
test4://
Exception in thread "main" java.lang.RuntimeException: 数组不符合规范!

ConstructBinaryTree

收获

  1.在递归问题中,代码可以用下标表示的就用下标表示,不用重新构建新的数组。

  2.数组为空与数组为null不是一回事。

更多:《剑指Offer》Java实现合集

【Java】 剑指offer(6) 重建二叉树的更多相关文章

  1. 剑指Offer:重建二叉树【7】

    剑指Offer:重建二叉树[7] 题目描述 输入某二叉树的前序遍历和中序遍历的结果,请重建出该二叉树.假设输入的前序遍历和中序遍历的结果中都不含重复的数字.例如输入前序遍历序列{1,2,4,7,3,5 ...

  2. 《剑指offer》重建二叉树

    本题来自<剑指offer> 重构二叉树 题目: 输入某二叉树的前序遍历和中序遍历的结果,请重建出该二叉树.假设输入的前序遍历和中序遍历的结果中都不含重复的数字.例如输入前序遍历序列{1,2 ...

  3. Go语言实现:【剑指offer】重建二叉树

    该题目来源于牛客网<剑指offer>专题. 输入某二叉树的前序遍历和中序遍历的结果,请重建出该二叉树.假设输入的前序遍历和中序遍历的结果中都不含重复的数字.例如输入前序遍历序列{1,2,4 ...

  4. 剑指offer之重建二叉树

    1.问题描述:输入某二叉树的前序遍历和中序遍历的结果,请重建出该二叉树.假设输入的前序遍历和中序遍历的结果中都不含重复的数字.        例如输入前序遍历序列pre {1,2,4,7,3,5,6, ...

  5. 剑指OFFER之重建二叉树(九度OJ1385)

    题目描述: 输入某二叉树的前序遍历和中序遍历的结果,请重建出该二叉树.假设输入的前序遍历和中序遍历的结果中都不含重复的数字.例如输入前序遍历序列{1,2,4,7,3,5,6,8}和中序遍历序列{4,7 ...

  6. 剑指offer:重建二叉树

    重建二叉树的前置知识: 0.遍历二叉树: (1)前序遍历:根左右 --> 先访问根节点,再前序遍历左子树,最后前序遍历右子树: (2)中序遍历:左根右 --> 先中序遍历左子树,再访问根节 ...

  7. 剑指Offer 4. 重建二叉树 (二叉树)

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

  8. 【剑指offer】重建二叉树

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

  9. 剑指offer——04重建二叉树(Python3)

    思路:在数据结构中,有一个条件反射,谈及二叉树,就递归.所以在实现重建二叉树时,也应该用到递归的思想. 在前序遍历中,根节点处于第一个:在中序遍历中,根节点的左边为左子树节点,根节点右边为右子树节点. ...

随机推荐

  1. json 不能 dumps Decimal 解决办法

    class DecimalEncoder(json.JSONEncoder): def default(self, o): if isinstance(o, decimal.Decimal): ret ...

  2. Mybatis进阶学习笔记——动态sql

    1.if标签 <select id="queryByNameAndTelephone" parameterType="Customer" resultTy ...

  3. C++学习1-(C语言基础、VS快捷键)

    C语言基础复习 1.三码 正数: 3码合1 ,正数的反码/补码就是其本身 负数: 原码就是符号位加上真值的绝对值,即用第一位表示符号,其余位表示值 原码:11010101 负数的反码是在其原码的基础上 ...

  4. ubuntu下tensorflow 报错 libcusolver.so.8.0: cannot open shared object file: No such file or directory

    解决方法1. 在终端执行: export LD_LIBRARY_PATH=”$LD_LIBRARY_PATH:/usr/local/cuda/lib64” export CUDA_HOME=/usr/ ...

  5. mysqlbinlog恢复数据注意事项【转】

    mysqlbinlog 恢复数据注意事项 前言: 上次有个有个朋友恢复 MySQL 数据,一直恢复不成功,也没有报错信息,使用的环境是 MySQL 5.7 使用了 GTID 以及 binlog 格式为 ...

  6. java并发编程系列七:volatile和sinchronized底层实现原理

    一.线程安全 1.  怎样让多线程下的类安全起来 无状态.加锁.让类不可变.栈封闭.安全的发布对象 2. 死锁 2.1 死锁概念及解决死锁的原则 一定发生在多个线程争夺多个资源里的情况下,发生的原因是 ...

  7. vs2010补丁

    背景 我都不知道这是干啥的了.想起.net真是一把鼻涕一把泪.搞了这么久.net,也被它坑了好多.有这时间搞搞开源东西多好.看见下面还有tfs,想起当时有个java同事竟然用vss管理java代码,后 ...

  8. (网络编程)socketserver模块服务端实现并发

    基于tcp的套接字(实现并发),关键就是两个循环,一个链接循环,一个通信循环 基于udp的套接字(不是正真意义上的并发,实现真并发) socketserver模块中分两大类:server类(解决链接问 ...

  9. Tpcc-MySQL对mysql数据库进行性能测试报告、分析及使用gnuplot生成图表展示

    TPC-C是专门针对联机交易处理系统(OLTP系统)的规范,一般情况下我们也把这类系统称为业务处理系统. tpcc-mysql是percona基于TPC-C(下面简写成TPCC)衍生出来的产品,专用于 ...

  10. Go语言规格说明书 之 类型(Types)

    go version go1.11 windows/amd64 本文为阅读Go语言中文官网的规则说明书(https://golang.google.cn/ref/spec)而做的笔记,完整的介绍Go语 ...