Given n, how many structurally unique BST's (binary search trees) that store values 1 ... n?html
Example:post
Input: 3 Output: 5 Explanation: Given n = 3, there are a total of 5 unique BST's: 1 3 3 2 1 \ / / / \ \ 3 2 1 1 3 2 / / \ \ 2 1 2 3
這道題其實是 卡塔蘭數 Catalan Numbe 的一個例子,若是對卡塔蘭數不熟悉的童鞋可能真不太好作。話說其實我也是今天才知道的好嘛 -.-|||,爲啥我之前都不知道捏?!爲啥卡塔蘭數不像斐波那契數那樣人盡皆知呢,是我太孤陋寡聞麼?!不過今天知道也不晚,不斷的學習新的東西,這纔是刷題的意義所在嘛! 好了,廢話很少說了,趕忙回到題目上來吧。咱們先來看當 n = 1 的狀況,只能造成惟一的一棵二叉搜索樹,n分別爲 1,2,3 的狀況以下所示:學習
1 n = 1 2 1 n = 2 / \ 1 2 1 3 3 2 1 n = 3 \ / / / \ \ 3 2 1 1 3 2 / / \ \ 2 1 2 3
就跟斐波那契數列同樣,咱們把 n = 0 時賦爲1,由於空樹也算一種二叉搜索樹,那麼 n = 1 時的狀況能夠看作是其左子樹個數乘以右子樹的個數,左右子樹都是空樹,因此1乘1仍是1。那麼 n = 2 時,因爲1和2均可覺得根,分別算出來,再把它們加起來便可。n = 2 的狀況可由下面式子算出(這裏的 dp[i] 表示當有i個數字能組成的 BST 的個數):url
dp[2] = dp[0] * dp[1] (1爲根的狀況,則左子樹必定不存在,右子樹能夠有一個數字)spa
+ dp[1] * dp[0] (2爲根的狀況,則左子樹能夠有一個數字,右子樹必定不存在)code
同理可寫出 n = 3 的計算方法:htm
dp[3] = dp[0] * dp[2] (1爲根的狀況,則左子樹必定不存在,右子樹能夠有兩個數字)blog
+ dp[1] * dp[1] (2爲根的狀況,則左右子樹均可以各有一個數字)ip
+ dp[2] * dp[0] (3爲根的狀況,則左子樹能夠有兩個數字,右子樹必定不存在)leetcode
由此能夠得出卡塔蘭數列的遞推式爲:
咱們根據以上的分析,能夠寫出代碼以下:
解法一:
class Solution { public: int numTrees(int n) { vector<int> dp(n + 1); dp[0] = dp[1] = 1; for (int i = 2; i <= n; ++i) { for (int j = 0; j < i; ++j) { dp[i] += dp[j] * dp[i - j - 1]; } } return dp[n]; } };
由卡特蘭數的遞推式還能夠推導出其通項公式,即 C(2n,n)/(n+1),表示在 2n 個數字中任取n個數的方法再除以 n+1,只要你尚未忘記高中的排列組合的知識,就不難寫出下面的代碼,注意在相乘的時候爲了防止整型數溢出,要將結果 res 定義爲長整型,參見代碼以下:
解法二:
class Solution { public: int numTrees(int n) { long res = 1; for (int i = n + 1; i <= 2 * n; ++i) { res = res * i / (i - n); } return res / (n + 1); } };
相似題目:
Different Ways to Add Parentheses
參考資料:
https://leetcode.com/problems/unique-binary-search-trees/