【Java】 剑指offer(6) 重建二叉树

【Java】 剑指offer(6) 重建二叉树

程序员徐师兄,六年大厂程序员,来源:https://github.com/gdutxiaoxu/Android_interview

本文参考自《剑指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:12473568//47215386
test2:12345//54321
test3:12345//12345
test4:1//1
Exception in thread "main" java.lang.RuntimeException: 数组不符合规范!

ConstructBinaryTree

收获

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

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

更多算法总结

《剑指Offer》整套汇总,都是用 Java 实现的 ,点击可以查看 Java 版本《剑指Offer》

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

打赏作者

IT徐师兄

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

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

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

打赏作者

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

抵扣说明:

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

余额充值