有 n 个气球,编号为0 到 n-1,每个气球上都标有一个数字,这些数字存在数组 nums 中。
现在要求你戳破所有的气球。每当你戳破一个气球 i 时,你可以获得 nums[left] * nums[i] * nums[right] 个硬币。 这里的 left 和 right 代表和 i 相邻的两个气球的序号。注意当你戳破了气球 i 后,气球 left 和气球 right 就变成了相邻的气球。
求所能获得硬币的最大数量。
说明:
你可以假设 nums[-1] = nums[n] = 1,但注意它们不是真实存在的所以并不能被戳破。
0 ≤ n ≤ 500, 0 ≤ nums[i] ≤ 100
示例:
输入: [3,1,5,8]
输出: 167
解释: nums = [3,1,5,8] --> [3,5,8] --> [3,8] --> [8] --> []
coins = 3*1*5 + 3*5*8 + 1*3*8 + 1*8*1 = 167
这个人的题解 简直太好了!!!,建议多看看
方法一 记忆化搜索
class Solution {
public:
int DFS(vector<int>& nums,int left ,int right,vector<vector<int>> &dp){
if(left == right-1)
return 0;
if(dp[left][right]!=-1)
return dp[left][right];
int Max=0;
for(int i=left+1;i<right;i++){
int tmp = DFS(nums,left,i,dp)+DFS(nums,i,right,dp) + nums[left]*nums[i]*nums[right];
if(tmp>Max)
Max = tmp;
}
dp[left][right] = Max;
return Max;
}
int maxCoins(vector<int>& nums) {
nums.insert(nums.begin(),1);
nums.push_back(1);
int len = nums.size();
vector<vector<int>> dp(len,vector<int>(len,-1));
return DFS(nums,0,len-1,dp);
}
};
方法二 Dp
class Solution {
public:
int maxCoins(vector<int>& nums) {
nums.insert(nums.begin(),1);
nums.push_back(1);
int len = nums.size();
vector<vector<int>> dp(len,vector<int>(len,0));
for(int l=2;l<=len;l++){
for(int i=0;i<=len-l;i++){
int j = i+l-1;
for(int k=i+1;k<j;k++)
dp[i][j] = max(dp[i][j],dp[i][k]+dp[k][j]+nums[i]*nums[k]*nums[j]);
}
}
return dp[0][len-1];
}
};