leetcode 刷题day41动态规划Part10(300.最长递增子序列、674. 最长连续递增序列、718. 最长重复子数组)

300.最长递增子序列

思路:根据递归五部曲进行分析。

1、dpi的定义

dpi表示i之前包括i的以numsi结尾的最长递增子序列的长度。

2、递推公式

位置i的最长升序子序列等于j从0到i-1各个位置的最长升序子序列 + 1 的最大值。

if (numsi > numsj) dpi = max(dpi, dpj + 1);

3、dpi的初始化

每一个i,对应的dpi(即最长递增子序列)起始大小至少都是1.

4、遍历顺序

dpi 是有0到i-1各个位置的最长递增子序列 推导而来,从前向后遍历。

j遍历0到i-1,前后遍历都可以,默认习惯 从前向后遍历。

5、举例推导dp数组

代码如下:

java 复制代码
class Solution {
    public int lengthOfLIS(int[] nums) {
        int[] dp=new int[nums.length];
        int result=1;
        for(int i=0;i<nums.length;i++) dp[i]=1;
        for(int i=1;i<nums.length;i++){
            for(int j=0;j<i;j++){
                if(nums[i]>nums[j]) dp[i]=Math.max(dp[i],dp[j]+1);
            }
            if(dp[i]>result) result=dp[i];
        }
        return result;
    }
}

674. 最长连续递增序列

思路:这个题目跟上一题的区别就在连续,所以只需要更改递推公式即可。

位置i的最长连续升序子序列等于i-1最长连续升序子序列 + 1 的最大值。

if (numsi > numsi-1) dpi = dpi-1 + 1;

代码如下:

java 复制代码
class Solution {
    public int findLengthOfLCIS(int[] nums) {
        int[] dp=new int[nums.length];
        int result=1;
        for(int i=0;i<nums.length;i++) dp[i]=1;
        for(int i=1;i<nums.length;i++){
            if(nums[i]>nums[i-1]) dp[i]=dp[i-1]+1;
            if(dp[i]>result) result=dp[i];
        } 
        return result;
    }
}

其实也可以暴力求解,计算每一个位置开始的最长连续递增序列长度,记录最长的结果。

718. 最长重复子数组

思路:定义二维数组dpij,i,j分别遍历A、B两个数组。

1、确定dp数组(dp table)以及下标的含义

dpij :以下标i - 1为结尾的A,和以下标j - 1为结尾的B,最长重复子数组长度为dpij。遍历dpij的时候i 和 j从1开始。

2、确定递推公式

根据dpij的定义,dpij的状态只能由dpi - 1j - 1推导出来。

即当Ai - 1 和Bj - 1相等的时候,dpij = dpi - 1j - 1 + 1;

3、dp数组如何初始化

根据dpij的定义,dpi0 和dp0j其实都是没有意义的。 为了方便递归公式dpij = dpi - 1j - 1 + 1,dpi0 和dp0j初始化为0。

举个例子A0如果和B0相同的话,dp11 = dp00 + 1,只有dp00初始为0,正好符合递推公式逐步累加起来。

4、确定遍历顺序

双层循环,都可以从前往后遍历。

5、举例推导dp数组

代码如下:

java 复制代码
class Solution {
    public int findLength(int[] nums1, int[] nums2) {
        int[][] dp=new int[nums1.length+1][nums2.length+1];
        int result=0;
        for(int i=1;i<=nums1.length;i++){
            for(int j=1;j<=nums2.length;j++){
                if(nums1[i-1]==nums2[j-1]) dp[i][j]=dp[i-1][j-1]+1;
                if(dp[i][j]>result) result=dp[i][j];
            }
        }
        return result;
    }
}

可以使用滚动数组将二维数组压缩为一维数组,这样内层循环需要从后往前遍历,以防覆盖。

java 复制代码
class Solution {
    public int findLength(int[] nums1, int[] nums2) {
        int[] dp=new int[nums2.length+1];
        int result=0;
        for(int i=1;i<=nums1.length;i++){
            for(int j=nums2.length;j>0;j--){
                if(nums1[i-1]==nums2[j-1]) dp[j]=dp[j-1]+1;
                else dp[j]=0;
                if(dp[j]>result) result=dp[j];
            }
        }
        return result;
    }
}
相关推荐
装不满的克莱因瓶6 分钟前
了解不同机器学习模型的分类
人工智能·python·算法·机器学习·ai·分类·数据挖掘
QiLinkOS8 分钟前
合肥气链科技有限公司本质总结
c++·科技·算法·gitee·开源
2501_9318037519 分钟前
线性筛(欧拉筛):从原理到应用
算法
ysu_031423 分钟前
leetcode数据结构与算法5~7:链表双指针与二级指针
数据结构·学习·算法·leetcode·链表
小欣加油26 分钟前
leetcode542 01矩阵
数据结构·c++·算法·leetcode·矩阵·bfs
wu_ye_m39 分钟前
学习c语言第34天 用函数每次输出+1,链式访问,int和void
c语言·学习·算法
星马梦缘1 小时前
算法设计与分析 作业三 答案与解析
算法·线性规划·二分图匹配·多元最短路·流网络·bellmanford·匈牙利树算法
微风欲寻竹影1 小时前
Java数据结构——二叉树(Binary Tree)详解
java·数据结构·算法
想吃火锅10051 小时前
【leetcode】3.无重复字符的最长字串js版
算法·leetcode·职场和发展
smith成长之旅1 小时前
08 | Mem0 框架分析: BM25 的 Sigmoid 归一化
数据库·python·算法