目录
引例
画图分析:
使用动态规划解决
1.状态表示
dp[i]表示以i位置元素为结尾的子序列中,最长递增子序列的长度
2.状态表示 对于子数组/子序列的问题都可以划分为长度为1,长度大于1两种情况
3.初始化 可以将dp表中的值初始化为最差的情况,即单个值构成子序列,初始化为1
4.填表顺序 从左往右
5.返回值 因为子序列的结束位置是任意的,所以返回值是dp表中的最大值
具体代码:
int lengthOfLIS(vector<int>& nums)
{
int n=nums.size();
vector<int> dp(n,1);
int ret=1;
for(int i=1;i<n;++i)
{
for(int j=0;j<i;++j)
if(nums[j]<nums[i]) dp[i]=max(dp[i],dp[j]+1);
ret=max(ret,dp[i]);
}
return ret;
}
经典LeetCode OJ题
1.第一题
画图分析:
使用动态规划解决
1.状态表示
dp[i]表示以i位置元素为结尾的所有子序列中,最长摆动子序列的长度
但在分析时,会发现结尾位置会出现两种情况
因此对于摆动的子序列或子数组,在状态表示时,是需要使用两个状态来表示的
f[i]表示以i位置元素为结尾的所有子序列中,最后一个位置呈现"上升"趋势的最长摆动序列的长度
f[i]表示以i位置元素为结尾的所有子序列中,最后一个位置呈现"下降"趋势的最长摆动序列的长度
2.状态转移方程----根据子序列构成来分析
3.初始化
可以将f,g表中的值先初始化为最差的情况,即1
4.填表顺序 从左往右两个表一起填
5.返回值 两个表中的最大值
具体代码:
int wiggleMaxLength(vector<int>& nums)
{
int n=nums.size();
vector<int> f(n,1),g(n,1);
int ret=1;
for(int i=1;i<n;++i)//填f[i],g[i]
{
for(int j=0;j<i;++j)
{
if(nums[j]<nums[i]) f[i]=max(g[j]+1,f[i]);
else if(nums[j]>nums[i]) g[i]=max(f[j]+1,g[i]);
}
ret=max(ret,max(f[i],g[i]));
}
return ret;
}
2.第二题
画图分析:
在使用动态规划解决该题之前,先介绍一个一次遍历求最大值及出现次数的小算法
使用动态规划解决
1.状态表示
dp[i]表示以i位置元素为结尾的所有子序列中,最长递增子序列的个数
但此时如果直接使用此状态表示求解状态转移方程的话,开始就会发现最长的递增子序列的长度是未知的,不能求解其次数,因此得使用两个状态表示来解决
len[i]表示以i位置元素为结尾的所有子序列中,最长递增子序列的长度
count[i]表示以i位置元素为结尾的所有子序列中,最长递增子序列的个数
2.状态转移方程
3.初始化 两个表都初始化为1
4.填表顺序 从左往右
5.返回值 使用小算法找到最长的子序列及出现次数
具体代码:
int findNumberOfLIS(vector<int>& nums)
{
int n=nums.size();
vector<int> len(n,1),count(n,1);
int retlen=1,retcount=1;
for(int i=1;i<n;++i)
{
for(int j=0;j<i;++j)
{
if(nums[j]<nums[i])
{
//说明是第二次出现,统计以j结尾的最长子序列的个数
if(len[j]+1==len[i]) count[i]+=count[j];
else if(len[j]+1>len[i]) len[i]=len[j]+1,count[i]=count[j];
}
}
if(retlen==len[i]) retcount+=count[i];
else if(retlen<len[i]) retlen=len[i],retcount=count[i];
}
return retcount;
}
3.第三题
画图分析:
使用动态规划解决
在使用动态规划解决之前,当以某个位置为结尾来研究问题时,是研究前面位置的状态,填表顺序是从左往右,但示例二在研究[7,8]进行填表时,会发现既可以跟在[1,2]后面,也可以跟在[4,5]后面,在填表时前后都会对其产生影响,因此要做预处理------排序
在做完排序后,就可以正常完成填表
1.状态表示
dp[i]表示以i位置元素为结尾的所有数对链中,最长数对链的长度
2.状态转移方程
3.初始化
一般将子序列或子数组问题dp表中的值,初始化为最差的情况,此题即1
4.填表顺序 从左往右
5.返回值 返回dp表中的最大值
具体代码:
int findLongestChain(vector<vector<int>>& pairs)
{
//排序预处理
sort(pairs.begin(),pairs.end());
int n=pairs.size();
vector<int> dp(n,1);
int ret=1;
for(int i=1;i<n;++i)
{
for(int j=0;j<i;++j)
{
if(pairs[j][1]<pairs[i][0]) dp[i]=max(dp[j]+1,dp[i]);
}
ret=max(ret,dp[i]);
}
return ret;
}
4.第四题
画图分析:
使用动态规划解决
1.状态表示
dp[i]表示以i位置元素为结尾的所有子序列中,最长的等差子序列的长度
2.状态转移方程
3.初始化
初始化第一个位置的值 hash[arr[0]]=1
4.填表顺序 从左往右
5.返回值 dp表中的最大值
具体代码:
int longestSubsequence(vector<int>& arr, int difference)
{
unordered_map<int,int> hash;//<arr[i],dp[i]>
hash[arr[0]]=1;//初始化
int ret=1;
for(int i=1;i<arr.size();++i)
{
hash[arr[i]]=hash[arr[i]-difference]+1;
ret=max(ret,hash[arr[i]]);
}
return ret;
}
5.第五题
OJ传送门 LeetCode<1218> 最长的斐波那契子序列的长度
画图分析:
使用动态规划解决
1.状态表示
dp[i]表示以i位置元素为结尾的所有子序列中,最长的斐波那契子序列的长度
如果使用此状态表示来研究状态转移方程的话,会发现i前面的j位置的dp[j]对应的arr[j]是可能跟在某个值后面的,就不能贸然使用dp[j]来更新dp[i],可以使用二维来确定
dp[i][j]表示以i位置及j位置元素为结尾的所有子序列中,最长的斐波那契额子序列的长度(i<j)
2.状态转移方程
3.初始化 表中所有值都初始化为2
此时可能会想,对于dp[0][0]初始化为2的话,根据状态表示是不正确的,但i<j,这就说明在填表时,只会用到上三角区域的值,因此这些值初始化为多少都不重要,是使用不到的
4.填表顺序 从上往下
5.返回值 dp表中的最大值 但此时得判断若数组是[1,2,4]是没有斐波那契子序列的,此时返回值ret的结果为2,根据题意是要为0的,因此需要做 ret<3? 0:ret;
具体代码:
int lenLongestFibSubseq(vector<int>& arr)
{
int n=arr.size();
//预处理使值与下标绑定
unordered_map<int,int> hash;//<value,index>
for(int i=0;i<n;++i) hash[arr[i]]=i;
vector<vector<int>> dp(n,vector<int>(n,2));
int ret=2;
for(int j=2;j<n;++j)//固定最后一个位置
{
for(int i=1;i<j;++i)//固定倒数第二个位置
{
int a=arr[j]-arr[i];
if(a<arr[i] && hash.count(a)) dp[i][j]=dp[hash[a]][i]+1;
ret=max(ret,dp[i][j]);
}
}
return ret<3? 0:ret;
}
6.第六题
画图分析:
使用动态规划解决
1.状态表示
dp[i]表示以i位置元素为结尾的所有子序列中,最长的等差子序列的长度
但在研究状态转移方程时,使用i位置前面的状态(dp[j])填dp[i]时,就会发现,我们只知道dp[j]是一个长度值,并不知道其具体的子序列的情况,可能在其后面跟nums[i]的话,构不成等差子序列,因此可以使用二维的来处理
dp[i][j]表示以i位置及j位置元素为结尾的所有子序列中,最长的等差序列的长度
2.状态转移方程
3.初始化
4.填表顺序
5.返回值 整个dp表中的最大值
具体代码:
int longestArithSeqLength(vector<int>& nums)
{
//优化
unordered_map<int,int> hash;
hash[nums[0]]=0;
int n=nums.size();
vector<vector<int>> dp(n,vector<int>(n,2));//创建dp表+初始化
int ret=2;
for(int i=1;i<n;++i)//固定倒数第二个数
{
for(int j=i+1;j<n;++j)//枚举倒数第一个位置
{
int a=2*nums[i]-nums[j];
if(hash.count(a) && hash[a]<i) dp[i][j]=dp[hash[a]][i]+1;
ret=max(ret,dp[i][j]);
}
hash[nums[i]]=i;
}
return ret;
}
7.第七题
OJ传送门 LeetCode<1218> 等差数列划分 II - 子序列
画图分析:
使用动态规划解决
1.状态表示
dp[i]表示以i位置元素为结尾的所有子序列中,等差子序列的个数
dp[i][j]表示以i位置及j位置元素为结尾的所有子序列中,等差子序列的个数(i<j)
2.状态转移方程
3.初始化
可以先全部初始化为最差的情况,即i,j单独构成一个子序列,但这不是一个等差序列,结果为0,因此可以先全部初始化为0
4.填表顺序
先固定倒数第一个数,再枚举倒数第二个数
5.返回值 整个dp表的和
具体代码:
int numberOfArithmeticSlices(vector<int>& nums)
{
//优化
int n=nums.size();
unordered_map<long long,vector<int>> hash;
for(int i=0;i<n;++i) hash[nums[i]].push_back(i);
int sum=0;
vector<vector<int>> dp(n,vector<int>(n));
for(int j=2;j<n;++j)//固定倒数第一个数,要使其有意义的话,从2开始
{
for(int i=1;i<j;++i)//固定倒数第二个数
{
long long a=(long long)2*nums[i]-nums[j];
if(hash.count(a))
{
for(auto k:hash[a])
{
if(k<i) dp[i][j]+=dp[k][i]+1;
}
}
sum+=dp[i][j];
}
}
return sum;
}