动态规划五步曲
一、整数拆分问题
思路:动态规划
-
-
- 确定递推公式
可以想 dp[i]最大乘积是怎么得到的呢?
其实可以从1遍历j,然后有两种渠道得到dp[i].
一个是j * (i - j) 直接相乘。
一个是j * dp[i - j],相当于是拆分(i - j),对这个拆分不理解的话,可以回想dp数组的定义。
那有人问了,j怎么就不拆分呢?
j是从1开始遍历,拆分j的情况,在遍历j的过程中其实都计算过了。
那么从1遍历j,比较(i - j) * j和dp[i - j] * j 取最⼤的。
递推公式:dp[i] = max(dp[i], max((i - j) * j, dp[i - j] * j));
-
- dp的初始化
不少同学应该疑惑,dp[0] dp[1]应该初始化多少呢?
有的题解里会给出dp[0] = 1,dp[1] = 1的初始化,但解释比较牵强,主要还是因为这么初始
化可以把题目过了。
严格从dp[i]的定义来说,dp[0] dp[1] 就不应该初始化,也就是没有意义的数值。
拆分0和拆分1的最大乘积是多少?
这是无解的。
这里只初始化dp[2] = 1,从dp[i]的定义来说,拆分数字2,得到的最大乘积是1。
-
- 确定遍历顺序
确定遍历顺序,先来看看递归公式:dp[i] = max(dp[i], max((i - j) * j, dp[i - j] * j));
dp[i] 是依靠 dp[i - j]的状态,所以遍历i⼀定是从前向后遍历,先有dp[i - j]再有dp[i]。
枚举j的时候,是从1开始的。i是从3开始,这样dp[i - j]就是dp[2]正好可以通过我们初始化的数值求出来。
-
5.举例推导dp数组
具体代码如下:
class Solution {
public:
int integerBreak(int n)
{
vector<int> dp(n+1);
dp[2] = 1;
for(int i = 3;i<=n;i++)
{
for(int j = 1;j<=i/2;j++)
{
dp[i] = max(dp[i],max(j*(i-j),j*dp[i-j]));
}
cout << dp[i] << " ";
}
return dp[n];
}
};
时间复杂度O(n^2),空间复杂度O(n)
总结
今天写了一道题目,整数拆分,仍然是关于动态规划的,我相信每天这么学一道题,收获一定会非常大!