LeetCode OJ 105. Construct Binary Tree from Preorder and Inorder Traversal

Given preorder and inorder traversal of a tree, construct the binary tree.

Note:
You may assume that duplicates do not exist in the tree.

通过树的中序序列和前序序列来构建一棵树。

例如:一棵树的前序序列是1-2-3,中序序列有5种可能,分别是1-2-3、2-1-3、1-3-2、2-3-1、3-2-1。不同的中序序列对应着不同的树的结构,这几棵树分别是[1,null,2,null,3]/[1,2,3]/[1,null,2,3]/[1,2,null,null,3]/[1,2,null,3]。

那么如何通过前序和中序序列来构建树呢?

我们知道:

1.前序序列是先visit根节点,然后visit左子树,最后visit右子树。

2.中序序列是先visit左子树,然后visit根节点,最后visit右子树。

因此,前序序列的第一个节点是根节点。我们在中序序列中找到根节点的位置,那么中序序列中根节点前面的节点就是根节点左子树中的节点,根节点后面的节点就是根节点右子树的节点。在上面的例子中,比如前序:1-2-3,中序2-1-3。由前序知道,1是根节点,在中序序列中找到1的位置,1前面的数是2,因此2是1的左子树,1的后面是3,因此3是1的右子树。所以对应是树为[1,2,3]。

知道了如何构建树,对应的代码如下:

 1 /**
 2  * Definition for a binary tree node.
 3  * public class TreeNode {
 4  *     int val;
 5  *     TreeNode left;
 6  *     TreeNode right;
 7  *     TreeNode(int x) { val = x; }
 8  * }
 9  */
10 public class Solution {
11     public TreeNode buildTree(int[] preorder, int[] inorder) {
12         if(preorder.length==0 || inorder.length==0) return null;
13         
14         TreeNode root = new TreeNode(preorder[0]);
15         int i;
16         for(i=0; i<inorder.length; i++){    //找到根节点在中序序列中的位置
17             if(inorder[i] == preorder[0]) break;
18         }
19         int[] left, right, npreorder;
20         if(i>0){
21             left = new int[i];              //左子树中的节点
22             System.arraycopy(inorder, 0, left, 0, left.length);
23         }else left = new int[0];
24         if(inorder.length - i -1 > 0){      //右子树中的节点
25             right = new int[inorder.length - i -1];
26             System.arraycopy(inorder, i+1, right, 0, right.length);
27         }else right = new int[0];
28         
29         npreorder = new int[preorder.length - 1];//删除根节点
30         System.arraycopy(preorder, 1, npreorder, 0, npreorder.length);
31         root.left = buildTree(npreorder, left);  //构建左子树
32         npreorder = new int[preorder.length - left.length - 1];//删除左子树中节点
33         System.arraycopy(preorder, 1 + left.length, npreorder, 0, npreorder.length);
34         root.right = buildTree(npreorder, right);//构建右子树
35         return root;
36     }
37 }
原文地址:https://www.cnblogs.com/liujinhong/p/5458643.html