312. 戳气球 (Hard)
问题描述
有 n
个气球,编号为 0
到 n - 1
,每个气球上都标有一个数字,这些数字存在数组 nums
中。
现在要求你戳破所有的气球。戳破第 i
个气球,你可以获得 nums[i - 1] * nums[i] * nums[i + 1]
枚硬币。 这里的 i - 1
和 i + 1
代表和 i
相邻的两个气球的序号。如果 i - 1
或 i + 1
超出了数组的边界,那么就当它是一个数字为 1
的气球。
求所能获得硬币的最大数量。
示例 1:
输入:nums = [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
示例 2:
输入:nums = [1,5]
输出:10
提示:
n == nums.length
1 <= n <= 300
0 <= nums[i] <= 100
解题思路
分治法+记忆化搜索
首先在数组的首尾各插入元素1
,以便于计算;
我们可以考虑将这个问题划分成两个不存在相互依赖的子问题,例如考虑[1, 3, 1, 5, 8, 1]
(注意首尾的1
是后来单独插入的),这里我们考虑,假设最后一个戳爆的气球是5
,那么该问题就可以划分成戳[1, 3, 1, 5]
和戳[5, 8, 1]
两组气球的得分再加上最后戳爆气球5
的得分,(注意这里数组中的第一个和最后一个元素实际上都不是气球的元素,不能戳)。
从递归的角度来说,有dfs(nums, left, right) = max(dfs(nums, left, mid) + dfs(nums, mid, right) + nums[mid] * nums[l] * nums[r])
;但是这样直接递归必然导致超时,因此我们考虑采用一个数组cach[left][right]
来保存dfs(nums, left, right)
的结果,如果cach[left][right]
不为0,说明dfs(nums, left, right)
已经计算过了,不需要再重复计算了,直接return cach[left][right]
就好了。
动态规划
由上面的记忆化搜索,其实可以非常方便地改写成动态规划的形式,即
dp[left][right] = max(dp[left][right], dp[left][mid] + dp[mid][right] + nums[left] * nums[mid] * nums[right])
;
这里状态dp[left][right]
的定义为:开区间(left, right)
引爆气球所能获得的最大分数,mid
表示的是最后一个引爆的气球的坐标;
因此,这里要注意状态转移方程遍历的顺序。
代码
分治法+记忆化搜索
class Solution {
public:
保证子问题之间不存在依赖, 分治:递归搜索+保存计算结果
int dfs(vector<int> &nums, int left, int right, vector<vector<int>> &cach) {
if (left >= right) {
return 0;
}
int maxnum = 0;
for (int i = left + 1; i < right; i++) {
if (cach[left][right] > 0)
return cach[left][right];
else
maxnum = std::max(maxnum, nums[left] * nums[i] * nums[right] + dfs(nums, left, i, cach) + dfs(nums, i, right, cach));
}
cach[left][right] = maxnum;
return maxnum;
}
int maxCoins(vector<int> &nums) {
nums.insert(nums.begin(), 1);
nums.push_back(1);
int n = nums.size();
vector<vector<int>> cach(n, vector<int>(n, 0));
return dfs(nums, 0, n - 1, cach);
}
}
动态规划
#include <vector>
using std::vector;
class Solution {
public:
int maxCoins(vector<int> &nums) {
nums.insert(nums.begin(), 1);
nums.push_back(1);
int n = nums.size();
vector<vector<int>> dp(n, vector<int>(n, 0));
for (int l = n - 1; l >= 0; l--) { // 注意这里的遍历顺序!
for (int r = 0; r < n; r++) {
for (int mid = l + 1; mid < r; mid++) {
dp[l][r] = max(dp[l][r], dp[l][mid] + dp[mid][r] + nums[mid] * nums[l] * nums[r]);
}
}
}
return dp[0][n - 1];
}
};
错误示范
class Solution {
public:
int dfs(vector<int> &nums, set<int> &rset, set<int, std::greater<int>> &lset, vector<vector<vector<int>>> &cach, int n) {
if (rset.size() == 2)
return 0;
int maxnum = 0; //应该说递归,转化为子问题的时候就出问题了,完全变成回溯了
for (int i = 1; i <= n; i++) {
auto iter = rset.find(i);
if (iter != rset.end()) {
int idx_left = *lset.upper_bound(i);
int idx_right = *rset.upper_bound(i);
if (cach[i][idx_left][idx_right] > 0) {
maxnum = std::max(maxnum, cach[i][idx_left][idx_right]);
} else {
rset.erase(i);
lset.erase(i);
cach[i][idx_left][idx_right] = nums[i] * nums[idx_left] * nums[idx_right] + dfs(nums, rset, lset, cach, n);
maxnum = std::max(maxnum, cach[i][idx_left][idx_right]);
rset.insert(i);
lset.insert(i);
}
}
}
return maxnum;
}
int maxCoins(vector<int> &nums) {
set<int> rset;
set<int, std::greater<int>> lset;
int n = nums.size();
nums.insert(nums.begin(), 1);
nums.push_back(1);
for (int i = 0; i < nums.size(); i++) {
rset.insert(i);
lset.insert(i);
}
vector<vector<vector<int>>> cach(nums.size(), vector<vector<int>>(nums.size(), vector<int>(nums.size(), -1)));
return dfs(nums, rset, lset, cach, n);
}
};
这份代码,如果去掉保存计算结果的数组,就变成暴力回溯了,那么答案正确,但是会超时,加上保存计算结果的数组之后,答案则会出现错误,cach[i][left][right] = nums[i] * nums[idx_left] * nums[idx_right] + dfs(nums, rset, lset, cach, n);
从表面含义来看,是在当前气球排列为left, i, right
的情况下,戳爆i
所能获得的最大硬币数,问题在于,这样写的到的结果是[0,1,2], [1,2,3], [2,3,4]...[i - 1, i, rihgt]
中的最大值。
312. 戳气球 (Hard)的更多相关文章
- Java实现 LeetCode 312 戳气球
312. 戳气球 有 n 个气球,编号为0 到 n-1,每个气球上都标有一个数字,这些数字存在数组 nums 中. 现在要求你戳破所有的气球.每当你戳破一个气球 i 时,你可以获得 nums[left ...
- Leetcode 312.戳气球
戳气球 有 n 个气球,编号为0 到 n-1,每个气球上都标有一个数字,这些数字存在数组 nums 中. 现在要求你戳破所有的气球.每当你戳破一个气球 i 时,你可以获得 nums[left] * n ...
- 312. 戳气球【困难】【区间DP】
题目链接 有 n 个气球,编号为0 到 n-1,每个气球上都标有一个数字,这些数字存在数组 nums 中. 现在要求你戳破所有的气球.每当你戳破一个气球 i 时,你可以获得 nums[left] * ...
- LeetCode 312. Burst Balloons(戳气球)
参考:LeetCode 312. Burst Balloons(戳气球) java代码如下 class Solution { //参考:https://blog.csdn.net/jmspan/art ...
- 312 Burst Balloons 戳气球
现有 n 个气球按顺序排成一排,每个气球上标有一个数字,这些数字用数组 nums 表示.现在要求你戳破所有的气球.每当你戳破一个气球 i 时,你可以获得 nums[left] * nums[i] * ...
- [Swift]LeetCode312. 戳气球 | Burst Balloons
Given n balloons, indexed from 0 to n-1. Each balloon is painted with a number on it represented by ...
- Q312 戳气球
有 n 个气球,编号为0 到 n-1,每个气球上都标有一个数字,这些数字存在数组 nums 中. 现在要求你戳破所有的气球.每当你戳破一个气球 i 时,你可以获得 nums[left] * nums[ ...
- leetcode 戳气球
有 n 个气球,编号为0 到 n-1,每个气球上都标有一个数字,这些数字存在数组 nums 中. 现在要求你戳破所有的气球.每当你戳破一个气球 i 时,你可以获得 nums[left] * nums[ ...
- Burst Balloons(leetcode戳气球,困难)从指数级时间复杂度到多项式级时间复杂度的超详细优化思路(回溯到分治到动态规划)
这道题目做了两个晚上,发现解题思路的优化过程非常有代表性.文章详细说明了如何从回溯解法改造为分治解法,以及如何由分治解法过渡到动态规划解法.解法的用时从 超时 到 超过 95.6% 提交者,到超过 9 ...
- leetcode312 戳气球
动态规划 time O class Solution { public: int maxCoins(vector<int>& nums) { nums.insert(nums.be ...
随机推荐
- ubuntu安装nvidia-docker2
1.配置源: distribution=$(. /etc/os-release;echo $ID$VERSION_ID) && curl -s -L https://nvidia.gi ...
- WinForms 嵌入 Web服务
1.首先安装一个Kestrel服务器包 Microsoft.AspNetCore.Server.Kestrel 2.在Main方法中插入如下代码 static class Program { /// ...
- VMvare虚拟机的安装及新建虚拟机(一)
a:hover { color: rgba(255, 102, 0, 1) } 一.VMvare虚拟机的安装 1.首先双击--你下载的安装包,这里我分享百度云盘,供大家下载:http://pan.ba ...
- 设置导航栏的title
self.navigationController.navigationBar.titleTextAttributes = [NSDictionary dictionaryWithObjectsAnd ...
- pyqt5离线安装教程
目前总结的安装pyqt5,需要的离线安装包,除了每一个包要跟系统版本适配之外,还要考虑包跟包之间的适配.pyqt5跟它开头的一些包要保持是同一个版本,至少有2个小数点的位数是一样的才行,qt5跟它开头 ...
- ElementUI Select下拉框定位问题!
今天遇到了下拉不跟随文本框滚动的问题 参考官方手册添加参数: popper-append-to-body="false" 无效[内心很无语]继续检查向上推,查看html样式,发现了 ...
- BT做种
BT方式:BT方式属于P2P传输,客户机之间可互传数据,大大提高传输效率,推荐使用.此处以qBittorrent做种为示例,主要有以下几个要点1.启用内置Tracker(或用其他同类软件代替)2.启用 ...
- 组件中的data为什么不是一个对象而是一个函数?
组件中的data为什么不是一个对象而是一个函数? 组件是可复用的vue实例,一个组件被创建好之后,就可能被用在各个地方,而组件不管被复用了多少次,组件中的data数据都应该是相互隔离,互不影响的,基于 ...
- java的jvm学习
- journalctl查看内核/应用日志
Systemd统一管理所有Unit的启动日志.带来的好处就是,可以只用journalctl一个命令,查看所有日志(内核日志和应用日志).日志的配置文件是/etc/systemd/journald.co ...