1. 程式人生 > >[Leetcode]Unique Binary Search Trees I & II

[Leetcode]Unique Binary Search Trees I & II

I

Given n, how many structurally unique BST's (binary search trees) that store values 1...n?

For example,
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

這一題用一維動態規劃填表法,解法很巧妙。

我們建立一個長度為n+1的count陣列,count[i]表示數0-i一共能產生多少個不同的BST。

這裡我們要深刻理解BST的一個性質,即BST的任意子樹依然是個BST

顯然的,我們可以發現:

count[0] = 1 空樹

count[1] = 1 只有1節點的樹

在觀察count[2],我們發現可以這樣計算count[2]:

選擇1做根節點, 由於BST的性質,其左子樹只能是空樹,只有count[0]種可能,右子樹是由節點2組成的樹,一共有count[1]種可能。

選擇2做根節點,同理可得左子樹有count[1]種,右子樹有count[0]種。

因此count[2] = count[0] * count[1] + count[1] * count[0]

通過這個分析,我們可以發現其實判斷有多少unique BST就是讓每個樹做根節點一次,計算他得左子樹數目和右子樹數目,然後相乘即可。

因此其Count[i] = ∑ Count[0...k] * [ k+1....i]

public int numTrees(int n) {
        int[] count = new int[n + 1];
        count[0] = 1;
        count[1] = 1;
        
        for (int i = 2; i <= n; i++) 
            for (int j = 0; j < i; j++) {
                count[i] += count[j] * count[i - 1 - j];
            }
        
        return count[n];
    }

II

Given n, generate all structurally unique BST's (binary search trees) that store values 1...n.

For example,
Given n = 3, your program should return all 5 unique BST's shown below.

   1         3     3      2      1
    \       /     /      / \      \
     3     2     1      1   3      2
    /     /       \                 \
   2     1         2                 3

II 的思路和I類似,用同樣的遞迴思路來做。

public List<TreeNode> generateTrees(int n) {
        return generateTrees(1, n);
    }
    
    private List<TreeNode> generateTrees(int start, int end) {
        List<TreeNode> list = new ArrayList<TreeNode>();
        if (start > end) {
            list.add(null);
            return list;
        }
        
        for (int i = start; i <= end; i++) {
            List<TreeNode> lefts = generateTrees(start, i - 1);
            List<TreeNode> rights = generateTrees(i + 1, end);
            for (TreeNode left: lefts) {
                for (TreeNode right: rights) {
                    TreeNode root = new TreeNode(i);
                    root.left = left;
                    root.right = right;
                    list.add(root);
                }
            }
        }
        return list;
    }