24年收尾之作------动态规划<六> 子序列问题(含对应LeetcodeOJ题)

目录

引例

[经典LeetCode OJ题](#经典LeetCode OJ题)

1.第一题

2.第二题

3.第三题

4.第四题

5.第五题

6.第六题

7.第七题

引例

OJ传送门 LeetCode<300>最长递增子序列

画图分析:

使用动态规划解决

1.状态表示

dpi表示以i位置元素为结尾的子序列中,最长递增子序列的长度

2.状态表示 对于子数组/子序列的问题都可以划分为长度为1,长度大于1两种情况

3.初始化 可以将dp表中的值初始化为最差的情况,即单个值构成子序列,初始化为1

4.填表顺序 从左往右

5.返回值 因为子序列的结束位置是任意的,所以返回值是dp表中的最大值

具体代码:

cpp 复制代码
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.第一题

OJ传送门 LeetCode<376>摆动序列

画图分析:

使用动态规划解决

1.状态表示

dpi表示以i位置元素为结尾的所有子序列中,最长摆动子序列的长度

但在分析时,会发现结尾位置会出现两种情况

因此对于摆动的子序列或子数组,在状态表示时,是需要使用两个状态来表示的

fi表示以i位置元素为结尾的所有子序列中,最后一个位置呈现"上升"趋势的最长摆动序列的长度

fi表示以i位置元素为结尾的所有子序列中,最后一个位置呈现"下降"趋势的最长摆动序列的长度

2.状态转移方程----根据子序列构成来分析

3.初始化

可以将f,g表中的值先初始化为最差的情况,即1

4.填表顺序 从左往右两个表一起填

5.返回值 两个表中的最大值

具体代码:

cpp 复制代码
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.第二题

OJ传送门 LeetCode<673>最长递增子序列的个数

画图分析:

在使用动态规划解决该题之前,先介绍一个一次遍历求最大值及出现次数的小算法

使用动态规划解决

1.状态表示

dpi表示以i位置元素为结尾的所有子序列中,最长递增子序列的个数

但此时如果直接使用此状态表示求解状态转移方程的话,开始就会发现最长的递增子序列的长度是未知的,不能求解其次数,因此得使用两个状态表示来解决

leni表示以i位置元素为结尾的所有子序列中,最长递增子序列的长度

counti表示以i位置元素为结尾的所有子序列中,最长递增子序列的个数

2.状态转移方程

3.初始化 两个表都初始化为1

4.填表顺序 从左往右

5.返回值 使用小算法找到最长的子序列及出现次数

具体代码:

cpp 复制代码
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.第三题

OJ传送门 LeetCode<646> 最长数对链

画图分析:

使用动态规划解决

在使用动态规划解决之前,当以某个位置为结尾来研究问题时,是研究前面位置的状态,填表顺序是从左往右,但示例二在研究7,8进行填表时,会发现既可以跟在1,2后面,也可以跟在4,5后面,在填表时前后都会对其产生影响,因此要做预处理------排序

在做完排序后,就可以正常完成填表

1.状态表示

dpi表示以i位置元素为结尾的所有数对链中,最长数对链的长度

2.状态转移方程

3.初始化

一般将子序列或子数组问题dp表中的值,初始化为最差的情况,此题即1

4.填表顺序 从左往右

5.返回值 返回dp表中的最大值

具体代码:

cpp 复制代码
 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.第四题

OJ传送门 LeetCode<1218> 最长定差子序列

画图分析:

使用动态规划解决

1.状态表示

dpi表示以i位置元素为结尾的所有子序列中,最长的等差子序列的长度

2.状态转移方程

3.初始化

初始化第一个位置的值 hasharr\[0]=1

4.填表顺序 从左往右

5.返回值 dp表中的最大值

具体代码:

cpp 复制代码
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.状态表示

dpi表示以i位置元素为结尾的所有子序列中,最长的斐波那契子序列的长度

如果使用此状态表示来研究状态转移方程的话,会发现i前面的j位置的dpj对应的arrj是可能跟在某个值后面的,就不能贸然使用dpj来更新dpi,可以使用二维来确定

dpij表示以i位置及j位置元素为结尾的所有子序列中,最长的斐波那契额子序列的长度(i<j)

2.状态转移方程

3.初始化 表中所有值都初始化为2

此时可能会想,对于dp00初始化为2的话,根据状态表示是不正确的,但i<j,这就说明在填表时,只会用到上三角区域的值,因此这些值初始化为多少都不重要,是使用不到的

4.填表顺序 从上往下

5.返回值 dp表中的最大值 但此时得判断若数组是1,2,4是没有斐波那契子序列的,此时返回值ret的结果为2,根据题意是要为0的,因此需要做 ret<3? 0:ret;

具体代码:

cpp 复制代码
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.第六题

OJ传送门 LeetCode<1218> 最长等差数列

画图分析:

使用动态规划解决

1.状态表示

dpi表示以i位置元素为结尾的所有子序列中,最长的等差子序列的长度

但在研究状态转移方程时,使用i位置前面的状态(dpj)填dpi时,就会发现,我们只知道dpj是一个长度值,并不知道其具体的子序列的情况,可能在其后面跟numsi的话,构不成等差子序列,因此可以使用二维的来处理

dpij表示以i位置及j位置元素为结尾的所有子序列中,最长的等差序列的长度

2.状态转移方程

3.初始化

4.填表顺序

5.返回值 整个dp表中的最大值

具体代码:

cpp 复制代码
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.状态表示

dpi表示以i位置元素为结尾的所有子序列中,等差子序列的个数

dpij表示以i位置及j位置元素为结尾的所有子序列中,等差子序列的个数(i<j)

2.状态转移方程

3.初始化

可以先全部初始化为最差的情况,即i,j单独构成一个子序列,但这不是一个等差序列,结果为0,因此可以先全部初始化为0

4.填表顺序

先固定倒数第一个数,再枚举倒数第二个数

5.返回值 整个dp表的和

具体代码:

cpp 复制代码
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;
    }
相关推荐
8Qi82 小时前
LeetCode 516:最长回文子序列
算法·leetcode·职场和发展·动态规划
wabs6668 小时前
关于动态规划【力扣63.不同路径II与62.不同路径的区别(C++)】自我总结
动态规划
三千里9 小时前
路径规划算法-备忘
算法·自动驾驶·动态规划
2601_9618451510 小时前
新高考一卷真题2025|真题PDF全科整理
线性代数·矩阵·pdf·动态规划·概率论·高考
随意起个昵称11 小时前
线性dp-LIS题目5(导弹拦截,二分优化)
c++·算法·动态规划
8Qi813 小时前
LeetCode 5:最长回文子串(Longest Palindromic Substring)—— 题解
算法·leetcode·职场和发展·动态规划
8Qi81 天前
LeetCode 1143 & 718:最长公共子序列 / 最长重复子数组
算法·leetcode·职场和发展·动态规划
随意起个昵称1 天前
线性dp-综合刷题1(Not Alone)
算法·动态规划
-森屿安年-1 天前
1137. 第 N 个泰波那契数
c++·动态规划
8Qi81 天前
LeetCode 115 & 392:不同子序列 / 判断子序列
算法·leetcode·职场和发展·动态规划