Given inorder and postorder traversal of a tree, construct the binary tree.html
Note:
You may assume that duplicates do not exist in the tree. 數組
這道題要求從中序和後序遍歷的結果來重建原二叉樹,咱們知道中序的遍歷順序是左-根-右,後序的順序是左-右-根,對於這種樹的重建通常都是採用遞歸來作,可參見我以前的一篇博客Convert Sorted Array to Binary Search Tree 將有序數組轉爲二叉搜索樹。針對這道題,因爲後序的順序的最後一個確定是根,因此原二叉樹的根節點能夠知道,題目中給了一個很關鍵的條件就是樹中沒有相同元素,有了這個條件咱們就能夠在中序遍歷中也定位出根節點的位置,並以根節點的位置將中序遍歷拆分爲左右兩個部分,分別對其遞歸調用原函數。代碼以下:函數
/** * Definition for binary tree * struct TreeNode { * int val; * TreeNode *left; * TreeNode *right; * TreeNode(int x) : val(x), left(NULL), right(NULL) {} * }; */ class Solution { public: TreeNode *buildTree(vector<int> &inorder, vector<int> &postorder) { return buildTree(inorder, 0, inorder.size() - 1, postorder, 0, postorder.size() - 1); } TreeNode *buildTree(vector<int> &inorder, int iLeft, int iRight, vector<int> &postorder, int pLeft, int pRight) { if (iLeft > iRight || pLeft > pRight) return NULL; TreeNode *cur = new TreeNode(postorder[pRight]); int i = 0; for (i = iLeft; i < inorder.size(); ++i) { if (inorder[i] == cur->val) break; } cur->left = buildTree(inorder, iLeft, i - 1, postorder, pLeft, pLeft + i - iLeft - 1); cur->right = buildTree(inorder, i + 1, iRight, postorder, pLeft + i - iLeft, pRight - 1); return cur; } };
上述代碼中須要當心的地方就是遞歸是postorder的左右index很容易寫錯,好比 pLeft + i - iLeft - 1, 這個又長又很差記,首先咱們要記住 i - iLeft 是計算inorder中根節點位置和左邊起始點的距離,而後再加上postorder左邊起始點而後再減1。咱們能夠這樣分析,若是根節點就是左邊起始點的話,那麼拆分的話左邊序列應該爲空集,此時i - iLeft 爲0, pLeft + 0 - 1 < pLeft, 那麼再遞歸調用時就會返回NULL, 成立。若是根節點是左邊起始點緊跟的一個,那麼i - iLeft 爲1, pLeft + 1 - 1 = pLeft,再遞歸調用時還會生成一個節點,就是pLeft位置上的節點,爲原二叉樹的一個葉節點。post
咱們下面來看一個例子, 某一二叉樹的中序和後序遍歷分別爲:ui
Inorder: 11 4 5 13 8 9url
Postorder: 11 4 13 9 8 5 spa
11 4 5 13 8 9 => 5code
11 4 13 9 8 5 / \htm
11 4 13 8 9 => 5blog
11 4 13 9 8 / \
4 8
11 13 9 => 5
11 13 9 / \
4 8
/ / \
11 13 9