Skip to content

Latest commit

 

History

History
 
 

956

Folders and files

NameName
Last commit message
Last commit date

parent directory

..
 
 

You are installing a billboard and want it to have the largest height.  The billboard will have two steel supports, one on each side.  Each steel support must be an equal height.

You have a collection of rods which can be welded together.  For example, if you have rods of lengths 1, 2, and 3, you can weld them together to make a support of length 6.

Return the largest possible height of your billboard installation.  If you cannot support the billboard, return 0.

 

Example 1:

Input: [1,2,3,6]
Output: 6
Explanation: We have two disjoint subsets {1,2,3} and {6}, which have the same sum = 6.

Example 2:

Input: [1,2,3,4,5,6]
Output: 10
Explanation: We have two disjoint subsets {2,3,5} and {4,6}, which have the same sum = 10.

Example 3:

Input: [1,2]
Output: 0
Explanation: The billboard cannot be supported, so we return 0.

 

Note:

  1. 0 <= rods.length <= 20
  2. 1 <= rods[i] <= 1000
  3. The sum of rods is at most 5000.

Related Topics:
Dynamic Programming

Solution 1. DP

Thought

For each rod x, we have 3 options:

  1. use it in left post
  2. use it in right post
  3. don't use it.

If we leave the numbers used in the left post as-is (x to +x), turn all the numbers used in the right post to negative (turn x to -x), and turn all numbers that are not used to 0, this problem becomes:

Find the max score we can get after doing the above operations. The "score" is the sum of all the positive numbers. For example, +1 +2 +3 -6 has a score of 6.

Since sum(A) is bounded, it suggests us to use this fact in some way.

A fact we should consider is that for a given sum, it doesn't matter how we get the sum.

For example, with A = [1,2,2,3], we could get sum 3 in 3 different ways. If we just consider sum = 3, we actually covered all those three cases.

Since sum is in range [-5000, 5000], we just have 10001 numbers to consider.

Algorithm

Let dp[i][s] be the largest score we can get after rewriting A[0..i] to get sum s. Our goal is get dp[N-1][0]. Note that in the implementation, we offset all the sums by 5000 for simplicity to avoid using negative sum as index. So, sum = 5000 actually means sum = 0.

For example, for A = [1,2,3,6], we have dp[2][2] = 4, because the highest score we can get after rewriting A[0..2] is 4 (sum = 2 = 1 - 2 + 3, score = 1 + 3 = 4).

For the base case, dp[-1][s] is 0 when s == 0, and -infinity everywhere else.

The recursion is:

dp[i][s] = max(
    dp[i-1][s], // write A[i] as 0
    A[i] + dp[i+1][s-A[i]], // write A[i] as +A[i]
    dp[i+1][s-A[i]]) // write A[i] as -A[i]
)
// OJ: https://leetcode.com/problems/tallest-billboard
// Author: github.com/lzl124631x
// Time: O(NS) where N is the length of `rods`,
//             and S is the maximum of `sum(rods[i..j])`
// Space: O(NS)
// Ref: https://leetcode.com/articles/tallest-billboard/
class Solution {
    vector<vector<int>> dp;
    int dfs(vector<int> &A, int i, int s) {
        if (i == -1) return s == 5000 ? 0 : INT_MIN;
        if (dp[i][s] != INT_MIN) return dp[i][s];
        int ans = dfs(A, i - 1, s); // Write A[i] as 0
        ans = max(ans, A[i] + dfs(A, i - 1, s - A[i])); // Write A[i] as A[i]
        ans = max(ans, dfs(A, i - 1, s + A[i])); // Write A[i] as -A[i]
        return dp[i][s] = ans;
    }
public:
    int tallestBillboard(vector<int>& A) {
        int N = A.size();
        dp = vector<vector<int>>(N, vector<int>(10001, INT_MIN));
        return dfs(A, N - 1, 5000);
    }
};