当前位置:网站首页>力扣题解 动态规划(4)
力扣题解 动态规划(4)
2022-07-27 05:20:00 【RL-UAV】
279.完全平方数
总结:
- 确定dp数组(dp table)以及下标的含义
dp[j]:和为j的完全平方数的最少数量为dp[j]
2.确定递推公式
dp[j] 可以由dp[j - i * i]推出, dp[j - i * i] + 1 便可以凑成dp[j]。
此时我们要选择最小的dp[j],所以递推公式:dp[j] = min(dp[j - i * i] + 1, dp[j]);
3.dp数组如何初始化
dp[0]表示 和为0的完全平方数的最小数量,那么dp[0]一定是0。
有同学问题,那0 * 0 也算是一种啊,为啥dp[0] 就是 0呢?
看题目描述,找到若干个完全平方数(比如 1, 4, 9, 16, …),题目描述中可没说要从0开始,dp[0]=0完全是为了递推公式。
非0下标的dp[j]应该是多少呢?
从递归公式dp[j] = min(dp[j - i * i] + 1, dp[j]);中可以看出每次dp[j]都要选最小的,所以非0下标的dp[j]一定要初始为最大值,这样dp[j]在递推的时候才不会被初始值覆盖。
4.确定遍历顺序
我们知道这是完全背包,
如果求组合数就是外层for循环遍历物品,内层for遍历背包。
dp[0] = 0 dp[1] = min(dp[0] + 1) = 1 dp[2] = min(dp[1] + 1) = 2 dp[3] = min(dp[2] + 1) = 3 dp[4] = min(dp[3] + 1, dp[0] + 1) = 1 dp[5] = min(dp[4] + 1, dp[1] + 1) = 2
错误:dp[i] = min(dp[i - j * j] + 1, dp[i]);
class Solution {
public:
int numSquares(int n) {
vector<int> dp(n + 1, INT_MAX);
dp[0] = 0;
for (int i = 0; i <= n; i++) {
// 遍历背包
for (int j = 1; j * j <= i; j++) {
// 遍历物品
dp[i] = min(dp[i - j * j] + 1, dp[i]);
}
}
return dp[n];
}
};
139.单词拆分
总结:
substr()是C++语言函数,主要功能是复制子字符串,要求从指定位置开始,并具有指定的长度。
class Solution {
public:
bool wordBreak(string s, vector<string>& wordDict) {
unordered_set<string> wordSet(wordDict.begin(), wordDict.end());
vector<bool> dp(s.size() + 1, false);
dp[0] = true;
for (int i = 1; i <= s.size(); i++) {
// 遍历背包
for (int j = 0; j < i; j++) {
// 遍历物品
string word = s.substr(j, i - j); //substr(起始位置,截取的个数)
if (wordSet.find(word) != wordSet.end() && dp[j]) {
dp[i] = true;
}
}
}
return dp[s.size()];
}
};
198.打家劫舍
总结:
dp[i]:考虑下标i(包括i)以内的房屋,最多可以偷窃的金额为dp[i]。
dp[i] = max(dp[i - 2] + nums[i], dp[i - 1]);
class Solution {
public:
int rob(vector<int>& nums) {
if (nums.size() == 0) return 0;
if (nums.size() == 1) return nums[0];
vector<int> dp(nums.size());
dp[0] = nums[0];
dp[1] = max(nums[0], nums[1]);
for (int i = 2; i < nums.size(); i++) {
dp[i] = max(dp[i - 2] + nums[i], dp[i - 1]);
}
return dp[nums.size() - 1];
}
};
213.打家劫舍II
总结:
而情况二 和 情况三 都包含了情况一了,所以只考虑情况二和情况三就可以了。
成环只要把单独的问题剥离出来,单独比较。\
错误: int result1 = robRange(nums, 0, nums.size() - 2);
nums.size() - 1
i <= end; 不是 i<end.
// 注意注释中的情况二情况三,以及把198.打家劫舍的代码抽离出来了
class Solution {
public:
int rob(vector<int>& nums) {
if (nums.size() == 0) return 0;
if (nums.size() == 1) return nums[0];
int result1 = robRange(nums, 0, nums.size() - 2); // 情况二
int result2 = robRange(nums, 1, nums.size() - 1); // 情况三
return max(result1, result2);
}
// 198.打家劫舍的逻辑
int robRange(vector<int>& nums, int start, int end) {
if (end == start) return nums[start];
vector<int> dp(nums.size());
dp[start] = nums[start];
dp[start + 1] = max(nums[start], nums[start + 1]);
for (int i = start + 2; i <= end; i++) {
dp[i] = max(dp[i - 2] + nums[i], dp[i - 1]);
}
return dp[end];
}
};
337.打家劫舍 III
总结:真难理解。
- 确定递归函数的参数和返回值
这里我们要求一个节点 偷与不偷的两个状态所得到的金钱,那么返回值就是一个长度为2的数组。
参数为当前节点,代码如下:
vector<int> robTree(TreeNode* cur) {
其实这里的返回数组就是dp数组。
所以dp数组(dp table)以及下标的含义:下标为0记录不偷该节点所得到的的最大金钱,下标为1记录偷该节点所得到的的最大金钱。
所以本题dp数组就是一个长度为2的数组!
如果是偷当前节点,那么左右孩子就不能偷,val1 = cur->val + left[0] + right[0]; (如果对下标含义不理解就在回顾一下dp数组的含义)
如果不偷当前节点,那么左右孩子就可以偷,至于到底偷不偷一定是选一个最大的,所以:val2 = max(left[0], left[1]) + max(right[0], right[1]);
最后当前节点的状态就是{val2, val1}; 即:{不偷当前节点得到的最大金钱,偷当前节点得到的最大金钱}
错误:返回根节点,不偷加偷
return {val2, val1};
class Solution {
public:
int rob(TreeNode* root) {
vector<int> result = robTree(root);
return max(result[0], result[1]);
}
// 长度为2的数组,0:不偷,1:偷
vector<int> robTree(TreeNode* cur) {
if (cur == NULL) return vector<int>{
0, 0};
vector<int> left = robTree(cur->left);
vector<int> right = robTree(cur->right);
// 偷cur
int val1 = cur->val + left[0] + right[0];
// 不偷cur
int val2 = max(left[0], left[1]) + max(right[0], right[1]);
return {
val2, val1};
}
};
- 时间复杂度:O(n),每个节点只遍历了一次
- 空间复杂度:O(\log n),算上递推系统栈的空间
边栏推荐
- Baiwen driver Daquan learning (I) LCD driver
- std::bind与std::function的一些应用
- [first song] deep learning of rebirth -keras (elementary)
- pytorch的多GPU训练的两种方式
- Gbase 8C - SQL reference 6 SQL syntax (14)
- operator() 用法之一
- [first song] rebirth of me in py introductory training (6): definition and application of functions
- 李宏毅 2020 深度学习与人类语言处理 DLHLP-Coreference Resolution-p21
- 超强远程连接管理工具:Royal TSX
- [MySQL learning] 8
猜你喜欢

pytorch的多GPU训练的两种方式

15. GPU acceleration, Minist test practice and visdom visualization

Live Home 3D Pro interior home design tool

arcgis for js api(2) 获取要素服务的id集合

AE 3D粒子系统插件:Trapcode Particular

QGIS系列(1)-QGIS(server-apache) win10安装

【Unity URP】代码获取当前URP配置UniversalRendererData,并动态添加RendererFeature

面试常问Future、FutureTask和CompletableFuture

【5·20特辑】MatLAb之我在和你表白

安全帽反光衣检测识别数据集和yolov5模型
随机推荐
C语言-动态内存管理
Live Home 3D Pro室内家居设计工具
个人开发者申请代码签名证书的签发流程
【头歌】重生之深度学习篇-Keras(初级)
代码随想录笔记_哈希_242有效的字母异位词
STM32 infrared remote control
韦东山 数码相框 项目学习(二)在LCD上显示中文字符
编程学习记录——第5课【分支和循环语句】
SoK: The Faults in our ASRs: An Overview of Attacks against Automatic Speech Recognition (题目过长)阅读笔记
Gbase 8C - SQL reference 6 SQL syntax (14)
cycleGAN解析
arcgis for js api(2) 获取要素服务的id集合
力扣第一周错题集
socket编程一:使用fork()实现最基础的并发模式
STM32-FSMC外扩内存SRAM
剪枝-量化-转onnx中文系列教程
[concurrent programming series 9] priorityblockingqueue, delayqueue principle analysis of blocking queue
开源WebGIS-相关知识
【头歌】重生之我在py入门实训中(12):Matplotlib接口和常用图形
11. Gradient derivation of perceptron