题目
剑指 Offer 07. 重建二叉树
输入某二叉树的前序遍历和中序遍历的结果,请重建该二叉树。假设输入的前序遍历和中序遍历的结果中都不含重复的数字。
例如,给出
前序遍历 preorder = [3,9,20,15,7]
中序遍历 inorder = [9,3,15,20,7]
返回如下的二叉树:
3
/ \
9 20
/ \
15 7
代码
/**
* Definition for a binary tree node.
* public class TreeNode {
* int val;
* TreeNode left;
* TreeNode right;
* TreeNode(int x) { val = x; }
* }
*/
class Solution {
int[] preorder;
//设定全局变量 前序变量数组
HashMap<Integer, Integer> dic = new HashMap<>();
//准备用于存储中序遍历顺序的哈希map
public TreeNode buildTree(int[] preorder, int[] inorder) {
this.preorder = preorder;
//全局变量会等于这个preorder
for (int i = 0; i < inorder.length; i++) {
dic.put(inorder[i], i);
//遍历存储inorder 注意Key是数组的值 后面value是位置
}
return recur(0, 0, inorder.length - 1);
//开始进行递归 根节点起初是0 左边界也是0 右边界是数组的长度-1
}
public TreeNode recur(int root, int left, int right) {
if (left > right) {
return null;
}
int i = dic.get(preorder[root]);
// i 是根节点在中序遍历中的位置
//经过这一步我们可以利用前序遍历的获得的根节点(在第一个) 来把中序遍历分成左右两端 中间就是根
TreeNode node = new TreeNode(preorder[root]);
node.left = recur(root + 1, left, i - 1);
//左循环根节点就是原来的根节点位置+1 左边界还是左边界 右边界是根节点的位置-1
node.right = recur(root + i - left + 1, i + 1, right);
//右循环根节点 = 根节点 + 左子树的长度 + 1
//左子树的长度 = 右边界 - 左边界 + 1
return node;
}
}