[LeetCode] Burst Balloons (Medium)
这题没有做出来. 自己的思路停留在暴力的解法, 时间复杂度很高:
- 初始化
maxCount = 0
. - 对于当前长度为
k
的数组nums
, 从0
到k - 1
逐个选取第i
个气球扎破- 计算扎破气球得到的金币数,
count = nums[i - 1] * nums[i] * nums[i + 1]
. - 从
nums
中删掉nums[i]
. - 查询
m[nums]
是否存在, 不存在则递归调用maxCoins(nums)
并插入m[nums]
. count += m[nums]
后就是本轮的最优解, 若count > maxCount
则更新maxCount
.- 恢复
nums[i]
回到nums
中.
- 计算扎破气球得到的金币数,
- 返回
maxCount
.
class Solution {
private:
map<vector<int>, int> m;
public:
int maxCoins(vector<int>& nums) {
int len = nums.size();
if (len <= 0) return 0;
if (len == 1) return nums[0];
int maxCount = 0;
for (int i = 0; i < len; ++i) {
int val = nums[i];
int count = nums[i];
if (i - 1 >= 0) {
count *= nums[i - 1];
}
if (i + 1 < len) {
count *= nums[i + 1];
}
nums.erase(nums.begin() + i);
auto it = m.find(nums);
if (it == m.end()) {
count += (m[nums] = maxCoins(nums));
} else {
count += it->second;
}
maxCount = max(maxCount, count);
nums.insert(nums.begin() + i, val);
}
return maxCount;
}
};
这个算法如果不加memo, 在第i
次选择时有n - i
个选择 (i = 0 ~ n-1)
, 所以一共有n!
次选择, 每次选择都要进行O(n)
时间复杂度的删气球和回填气球的操作查找vector
的时间复杂度是多少?, 故时间复杂度O(n*n!)
, n
层递归故空间复杂度O(n)
.
如上加了memo之后, 对于每种输入情况只计算一次, 对于k
个气球, 有C(n, k)
中情况 (k = 1~n)
, 一共是2^n
种情况, 同上要考虑增删气球的时间复杂度, 所以时间复杂度最好也是O(n*2^n)
. 这2^n
种情况都要保存在memo中, 每种输入的平均长度是O(n)
级别的, 因此空间复杂度是O(n*2^n)
.
总之, TLE.
看了Share some analysis and explanations之后写下了下面的算法.
class Solution {
private:
map<pair<int, int>, int> m;
int maxCoins(vector<int> &nums, int left, int right, int lv, int rv) {
if (left > right) return 0;
if (left == right) return nums[left] * lv * rv;
auto coor = make_pair(left, right);
auto it = m.find(coor);
if (it != m.end()) {
return it->second;
}
int maxCount = 0;
for (int i = left; i <= right; ++i) {
int count = nums[i] * lv * rv
+ maxCoins(nums, left, i - 1, lv, nums[i])
+ maxCoins(nums, i + 1, right, nums[i], rv);
maxCount = max(maxCount, count);
}
m[coor] = maxCount;
return maxCount;
}
public:
int maxCoins(vector<int>& nums) {
int len = nums.size();
if (len <= 0) return 0;
return maxCoins(nums, 0, len - 1, 1, 1);
}
};
// Runtime: 1428ms
时间复杂度O(n^3)
. 起止点共有C(n, 2)
个组合, 是O(n^2)
级别的. 对于每个组合要遍历一遍, 找最大. 所以整体是O(n^3)
.
空间复杂度O(n^2)
.
小优化:
- 删掉0. (Runtime: 1300ms).
- 改用数组做map (Runtime: 36ms). 没想到数组比map好用这么多, 为什么?
class Solution {
private:
int maxCoins(vector<int> &nums, int left, int right, int lv, int rv, int* memo, int n) {
if (left > right) return 0;
if (left == right) return nums[left] * lv * rv;
if (memo[left * n + right] != 0) return memo[left * n + right];
int maxCount = 0;
for (int i = left; i <= right; ++i) {
int count = nums[i] * lv * rv
+ maxCoins(nums, left, i - 1, lv, nums[i], memo, n)
+ maxCoins(nums, i + 1, right, nums[i], rv, memo, n);
maxCount = max(maxCount, count);
}
memo[left * n + right] = maxCount;
return maxCount;
}
public:
int maxCoins(vector<int>& nums) {
int len = nums.size();
if (len <= 0) return 0;
int n = 0;
for (int x : nums) if (x > 0) nums[n++] = x;
int memo[n][n] = {};
return maxCoins(nums, 0, n - 1, 1, 1, (int*)memo, n);
}
};
标准答案:
class Solution {
public:
int maxCoins(vector<int>& iNums) {
int nums[iNums.size() + 2];
int n = 1;
for (int x : iNums) if (x > 0) nums[n++] = x;
nums[0] = nums[n++] = 1;
int dp[n][n] = {};
for (int k = 2; k <= n; ++k) {
for (int L = 0; L <= n - k; ++L) {
int R = L + k;
for (int i = L + 1; i < R; ++i) {
dp[L][R] = max(dp[L][R], nums[L] * nums[i] * nums[R] + dp[L][i] + dp[i][R]);
}
}
}
return dp[0][n - 1];
}
};
// Runtime: 12ms
其中dp[i][j]
表示第i
个气球到第j
个气球能获取的最大金币数 (i
最小为-1
, j
最大为n
).
[LeetCode] Burst Balloons (Medium)的更多相关文章
- [LeetCode] Burst Balloons 打气球游戏
Given n balloons, indexed from 0 to n-1. Each balloon is painted with a number on it represented by ...
- LeetCode Burst Balloons
原题链接在这里:https://leetcode.com/problems/burst-balloons/ 题目: Given n balloons, indexed from 0 to n-1. E ...
- LeetCode 312. Burst Balloons(戳气球)
参考:LeetCode 312. Burst Balloons(戳气球) java代码如下 class Solution { //参考:https://blog.csdn.net/jmspan/art ...
- 【LeetCode】452. Minimum Number of Arrows to Burst Balloons 解题报告(Python)
[LeetCode]452. Minimum Number of Arrows to Burst Balloons 解题报告(Python) 标签(空格分隔): LeetCode 题目地址:https ...
- LN : leetcode 312 Burst Balloons
lc 312 Burst Balloons 312 Burst Balloons Given n balloons, indexed from 0 to n-1. Each balloon is pa ...
- 贪心:leetcode 870. Advantage Shuffle、134. Gas Station、452. Minimum Number of Arrows to Burst Balloons、316. Remove Duplicate Letters
870. Advantage Shuffle 思路:A数组的最大值大于B的最大值,就拿这个A跟B比较:如果不大于,就拿最小值跟B比较 A可以改变顺序,但B的顺序不能改变,只能通过容器来获得由大到小的顺 ...
- 动态规划-Burst Balloons
Burst Balloons Given n balloons, indexed from 0 to n-1. Each balloon is painted with a number on it ...
- [LeetCode] Minimum Number of Arrows to Burst Balloons 最少数量的箭引爆气球
There are a number of spherical balloons spread in two-dimensional space. For each balloon, provided ...
- [LeetCode] 312. Burst Balloons 打气球游戏
Given n balloons, indexed from 0 to n-1. Each balloon is painted with a number on it represented by ...
随机推荐
- 利用javascript实现文本的自动输出
主要利用了setTimeout(),递归和String.substring(); 做出的效果就像是有一个打字员在打字. <!doctype html> <html lang=&quo ...
- java 静态方法
在使用java的时候,你会发现,有些对象,需要new ,有些则不需要时,比如Math类 ); 如果你查看源码就会大致的发现,里面的属性和方法都是静态的: public static double si ...
- LSJ_NHibernate第三章 IDAL,DAL,BLL
前言: 做项目(面向数据编程),首先必须了解业务,这是核心,不懂业务写出来的代码毫无意义.业务我这里分为两种,简单业务操作,复杂业务操作,我以他们操作表的界限进行区分,假设我更新一条数据,只操作了一张 ...
- 达夫设备/达夫算法(Duff's Device)
主要是下面的代码: register n = (count + 7) / 8; /\* count > 0 assumed \*/ switch (count % 8) { case 0: ...
- Timestamp的作用及与字符串的相互转换 .
一.Timestamp的介绍 每一个数据库都有一个计数器,这个计数器记录着数据行的插入.更新行为.如果我们为一个表中增加 timestamp 列,那么,该列将记录每一个数据行的计数器值.假如数据库中当 ...
- Eclipse 配置SSH 详解
http://blog.csdn.net/binyao02123202/article/details/18446523 最近看了很多招聘,其中很多我想去的公司都需要一些技能,其中熟练 Java SS ...
- 【转】浅析Windows编程的剪贴板
摘要: 本文对Windows剪贴板机制作了深入.全面的阐述,具体内容包括:文本.位图.DSP.自定义格式剪贴板的使用和多数据项和延迟提交技术. 关键词: VC++6.0: 剪贴板机制:数据格式:延迟提 ...
- 学习笔记_Java_day13_JSP三大指令()
JSP指令 1 JSP指令概述 JSP指令的格式:<%@指令名 attr1=”” attr2=”” %>,一般都会把JSP指令放到JSP文件的最上方,但这不是必须的. JSP ...
- dbms_job涉及到的知识点
用于安排和管理作业队列,通过使用作业,可以使ORACLE数据库定期执行特定的任务. 一.dbms_job涉及到的知识点1.创建job:variable jobno number;dbms_job.su ...
- IOS 学习笔记 2015-03-20 OC-集合-数组
[NSArray] 一 定义 1 不可变数组 2 oc中数组的元素可以是任何对象 3 数字中装有元素的地址 二 初始化 NSArray *变量 = [[NSArry alloc] initWithOb ...