【Leetcode公共子序列】718.最长重复子数组(子序列连续)1143.最长公共子序列(子序列不连续) 1035.不相交的线,392. 判断子序列(子序列不连续,这两道题和1143一摸一样)

文章目录



总结

(一)718连续的dp定义和递推:

定义dp[i][j]为以下标 i-1 为结尾的A,和以下标 j-1 为结尾的B,最长重复子数组长度

if(nums1[i-1]==nums2[j-1]) dp[i][j]=dp[i-1][j-1]+1;
//if(nums1[i]!=nums2[j]) dp[i][j]=max(dp[i][j-1],dp[i-1][j]);

(二)1143不连续的dp定义和递推:

定义dp[i][j]为,[0, i-1] 范围内的A,和[0, j-1] 范围内的的B,最长重复子序列长度

if(text1[i-1]==text2[j-1]) dp[i][j]=dp[i-1][j-1]+1;
else dp[i][j]=max(dp[i-1][j],dp[i][j-1]);

(三)区别:

1.dp[i][j]定义一个是必须以i,j结尾,一个是范围

2.递推一个只能由dp[i-1][j-1]推出,一个还有dp[i-1][j]和dp[i][j-1]

3.并且由于dp定义不同dp数组大小自然也不同,一个是len+1,一个是len

(四)不连续->[0,i-1]范围->两个状态来源->答案为dp[len1][len2]; 连续->以i-1结尾->一个状态来源->答案为所有dp数组的最大值;



Leetcode718(要求公共子序列连续)

1.问题描述

在这里插入图片描述

2.解决方案


递推很重要!!!!

这道题的递推只有上面那行,因为dp[i][j]的状态只能由dp[i-1][j-1]推导出,由于定义是以i,j为结尾的最长,而不是[0,i][0,j]范围内的最长

if(nums1[i]==nums2[j]) dp[i][j]=dp[i-1][j-1]+1;
//if(nums1[i]!=nums2[j]) dp[i][j]=max(dp[i][j-1],dp[i-1][j]);


解法一:自己的dp定义,二维数组

自己的dp定义:定义dp[i][j]为以下标 i 为结尾的A,和以下标 j 为结尾的B,最长重复子数组长度

1.这样的定义难点在于初始化,相对复杂,需要对第0行,第0列进行初始化
2.并且需要用flag标记第0行,第0列是否有1,有1,在循环前就得记录

class Solution {
public:
    int findLength(vector<int>& nums1, vector<int>& nums2) {
        //1.
        int len1=nums1.size();
        int len2=nums2.size();

        //2.
        vector<vector<int> > dp(len1,vector<int>(len2,0));
        bool flag= false;
        for(int i=0;i<len1;i++){
            if(nums1[i]==nums2[0]) {dp[i][0]=1; flag=true;}
         }
        for(int i=0;i<len2;i++){
            if(nums2[i]==nums1[0]) {dp[0][i]=1; flag=true;}
        }


        //3.
        int ans=0;
        if(flag== true) ans=1;
        for(int i=1;i<len1;i++){
            for(int j=1;j<len2;j++){
                if(nums1[i]==nums2[j]) dp[i][j]=dp[i-1][j-1]+1;
                //if(nums1[i]!=nums2[j]) dp[i][j]=max(dp[i][j-1],dp[i-1][j]);
                ans=max(ans,dp[i][j]);
            }
        }

        //4.
        return ans;
    }
};


解法二:代码sxl实现合理的dp定义,二维数组

代码sxl合理的dp定义:定义dp[i][j]为以下标 i-1 为结尾的A,和以下标 j-1 为结尾的B,最长重复子数组长度

1.初始化很简单,根绝定义和递推需要全部为0
2.递推一样,只有一种状态来源

class Solution1 {
public:
    int findLength(vector<int>& nums1, vector<int>& nums2) {
        //1.
        int len1=nums1.size();
        int len2=nums2.size();

        //2.
        vector<vector<int> > dp(len1+1,vector<int>(len2+1,0));
        //bool flag= false;
        //for(int i=0;i<len1;i++){
        //    if(nums1[i]==nums2[0]) {dp[i][0]=1; flag=true;}
        //}
        //for(int i=0;i<len2;i++){
        //    if(nums2[i]==nums1[0]) {dp[0][i]=1; flag=true;}
        //}


        //3.
        int ans=0;
        //if(flag== true) ans=1;
        for(int i=1;i<=len1;i++){
            for(int j=1;j<=len2;j++){
                if(nums1[i-1]==nums2[j-1]) dp[i][j]=dp[i-1][j-1]+1;
                //if(nums1[i]==nums2[j]) dp[i][j]=dp[i-1][j-1]+1;
                //if(nums1[i]!=nums2[j]) dp[i][j]=max(dp[i][j-1],dp[i-1][j]);
                ans=max(ans,dp[i][j]);
            }
        }

        //4.
        return ans;
    }
};


解法三:代码sxl实现合理的dp定义,二维数组滚动数组

1.内层只用上一层的数据,不用新更新的数据,所以内层倒序遍历
2.并且如果不发生状态转移要归零,方便下一层使用

class Solution {
public:
    int findLength(vector<int>& nums1, vector<int>& nums2) {
        //1.
        int len1=nums1.size();
        int len2=nums2.size();

        //2.
        vector<int> dp(len2+1,0);


        //3.
        int ans=0;
        for(int i=1;i<=len1;i++){
            for(int j=len2;j>=1;j--){
                if(nums1[i-1]==nums2[j-1]) dp[j]=dp[j-1]+1;
                else dp[j]=0;
                ans=max(ans,dp[j]);
            }
        }

        //4.
        return ans;
    }
};



Leetcode1143(要求公共子序列不一定连续)

1.问题描述

在这里插入图片描述

2.解决方案

a.本题和718除了要求子序列不连续其他的都一样

b.dp的定义由以i-1 j-1 结尾改为 [0,i-1][0,j-1] 范围内的公共子序列

c.定义不同递推也增加一条dp[i][j]=max(dp[i-1][j],dp[i][j-1]);

d.定义不是范围[0,i] [0,j],而是范围[0,i-1] [0,j-1],所以全部初始化为0就好

class Solution {
public:
    int longestCommonSubsequence(string text1, string text2) {
        //1.
        int len1=text1.size();
        int len2=text2.size();
        if(len1==0||len2==0) return 0;

        //2.
        vector<vector<int> > dp(len1+1,vector<int>(len2+1,0));

        //3.
        for(int i=1;i<=len1;i++){
            for(int j=1;j<=len2;j++){
                if(text1[i-1]==text2[j-1]) dp[i][j]=dp[i-1][j-1]+1;
                else dp[i][j]=max(dp[i-1][j],dp[i][j-1]);
            }
        }

        //4.
        return dp[len1][len2];

    }
};



Leetcode1035(要求公共子序列不一定连续,和1143一摸一样)

1.问题描述

在这里插入图片描述

2.解决方案

这道题目首先你得把题目分析清楚了,要连线那就是要求求完全相同的子序列,连线不能相交那就保持原来的顺序,那这么一分析发现和1143是一摸一样的求不连续最长子序列长度,所以代码一摸一样不用动!

class Solution {
public:
    int maxUncrossedLines(vector<int>& nums1, vector<int>& nums2) {
        //1.
        int len1=nums1.size();
        int len2=nums2.size();

        //2.
        vector<vector<int> > dp(len1+1,vector<int>(len2+1,0));

        //3.
        for(int i=1;i<=len1;i++){
            for(int j=1;j<=len2;j++){
                if(nums1[i-1]==nums2[j-1]) dp[i][j]=dp[i-1][j-1]+1;
                else dp[i][j]=max(dp[i-1][j],dp[i][j-1]);
            }
        }

        //4.
        return dp[len1][len2];
    }
};



Leetcode392(要求公共子序列不一定连续,和1143一摸一样)

1.问题描述

在这里插入图片描述

2.解决方案

a.本题完全符合上面总结的不连续的套路,dp定义是范围,状态两个来源,答案是dp[len1][len2]
b.本题稍微一转化就发现和1143是一模一样的,就是求不连续公共子序列最长长度,只不过这道题,最后只需要判断不连续公共子序列最长长度是不是等于s.size(),等于那就是子序列不等于那就不是,其他的一摸一样,包括dp定义,递推,初始化等等

class Solution {
public:
    bool isSubsequence(string s, string t) {
        //1.
        int len1=s.size();
        int len2=t.size();

        //2.
        vector<vector<int> > dp(len1+1,vector<int>(len2+1,0));
        
        //3.
        for(int i=1;i<=len1;i++){
            for(int j=1;j<=len2;j++){
                if(s[i-1]==t[j-1]) dp[i][j]=dp[i-1][j-1]+1;
                else dp[i][j]=max(dp[i-1][j],dp[i][j-1]);
            }
        }
        
        //4.
        if(dp[len1][len2]==len1) return true;
        else return false;
    }
};
  • 2
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
LeetCode总结 1. 数组 1.1 从有序数组中删除重复元素 1.2 在排序数组被旋转后进行查找 1.3 寻找两个排序数组的中位数 1.4 最长连续序列 1.5 累加和 1.6 移除数组中指定值 1.7 下一个排列 1.8 第n个全排列 1.9 验证数独的正确性 1.10 容纳雨水的量 1.11 旋转图像 1.12 数字加1 1.13 爬楼梯 1.14 格雷码 1.15 设置矩阵的行列为0 1.16 加油站问 1.17 分糖果 1.18 只出现一次的数 2. 单链表 2.1 单链表相加 2.2 指定位置反转单链表 2.3 依据给定值将链表重新排序 2.4 删除链表中重复元素 2.5 指定位置旋转链表 2.6 删除倒数第N个节点 2.7 成对交换链表元素 2.8 复制复杂链表 2.9 链表环相关问 2.9.1 链表是否有环 2.9.2 链表环的入口 2.10 改变链表中的元素位置2.11 LRU Cache(设计) 3. 字符串 3.1 判断字符串是否为回文 3.2 实现strStr() 3.3 字符串转为int(atoi) 3.4 二进制树相加 3.5 最长回文字符串 3.6 正则表达式匹配[hard] 3.7 正则匹配 3.8 最长公共前缀 3.9 验证字符串是否为数字 3.10 数字转为罗马数字 3.11 罗马数字到数字 3.12 Count and Say 3.13 变位词 3.14 简化系统路径 3.15 最后一个单词的长度 3.16 反转字符串中的单词 3.16.1 字符串前后和中间可能存在多个空格 3.16.2 不存在前后和中间的多余空格 3.17 一个编辑距离 4. 栈 4.1 验证括号的正确性 4.2 最长的正确括号表达式 4.3 柱状图中的最大矩形面积 4.4 计算逆波兰式的值 5. 树 5.1 二叉树的遍历 5.1.1 二叉树的前、中、后序遍历 5.1.2 二叉树的层序遍历 5.1.3 恢复二叉树[hard] 5.1.4 判断两棵树是否相等 5.1.5 判断二叉树是否为AVL树 5.1.6 将二叉树转为链表 5.1.7 二叉树添加指向右边节点的指针 5.1.8 树中节点的最小公共祖先 5.2 二叉树的构建5.3 二叉查找树 5.3.1 生成不重复的二叉查找树数目 5.3.2 验证是否为二叉查找树 5.3.3 将有序数组转为二叉树 5.3.4 将有序链表转为二叉树 5.4 二叉树的递归 5.4.1 二叉树的最大深度 5.4.2 二叉树的最小深度 5.4.3 路径和 5.4.4 满二叉树添加指向右边节点的指针 5.4.5 根节点到叶结点的所有路径代表的数字之和 6. 排序 6.1 合并两个有序数组到其中一个数组 6.2 合并两个有序链表 6.3 合并K个有序链表 6.4 使用插入排序来排序链表 6.5 归并排序排序链表 6.6 第一个缺少的正数 6.7 排序颜色 7. 查找 7.1 在排序数组中查找数出现的范围 7.2 在排序数组中查找给定值的插入位置 7.3 在二维排序数组中查找给定值 7.4 在旋转有序数组中查找最小值 7.4.1 数组无重复 7.4.2 数组有重复 7.5 在旋转排序数组中查找指定数字 8. 暴力枚举法 8.1 求集合的子集 8.2 集合的全排列 8.3 在指定树中选择进行全排列 8.4 电话上对应数字的字母组成的所有单词 9. 广度优先搜索 9.1 单词变换路径(Word Ladder) 9.1.1 是否存在变换路径 9.1.2 所有最短变换路径9.2 包围区域 10. 深度优先搜索 10.1 N皇后问 10.2 恢复IP地址 10.3 集合元素之和 10.3.1 元素可以重复 10.3.2 元素不可重复 10.3.3 给定元素数目和元素范围 10.4 正确的括号对 10.5 解数独 10.6 单词搜索 10.7 小结 10.7.1 适用场景 10.7.2 思考步骤 10.7.3 代码模板 10.7.4 深搜与回溯、递归的区别 11. 分治法 11.1 实现pow(x, n) 11.2 Sqrt(x) 12. 贪心算法 12.1 跳台阶游戏 12.2 买卖股票的最佳时机 12.2.1 最多允许交易一次 12.2.2 可以交易任意多次 12.2.3 最多可以交易两次 12.2.4 可以交易任意多次 12.2.5 交易后要停止一段时间 12.3 最长不含重复元素的子串 12.4 存放的最大水量 13. 动态规划 13.1 三角形从顶到底的最小路径和 13.2 最大连续数组 13.3 字符串的所有子回文字符串 13.4 最长公共子序列 13.5 字符串的编辑距离 13.6 不同路径之和 13.6.1 无障碍13.6.2 有障碍 13.7 最大矩形面积 13.8 字符串交叉组合 13.9 旋转字符串 13.10 最小路径和 13.11 所有的编码方式 13.12 独一无二的子序列数 13.13 拆分单词 13.13.1 单词是否由词典中的单词组成 13.13.2 返回所有可以切分的解 14. 图 14.1 图的克隆 15. 细节实现 15.1 反转整数 15.2 对称数判断 15.3 区间的相关操作 15.3.1 在区间中插入新的区间 15.3.2 合并区间 15.4 包含子串元素的最小窗口 15.5 大数乘法 15.6 给定串中是否存在包含所有单词的子串 15.7 Pascal 三角形 15.7.1 生成Pascal三角形 15.7.2 Pascal三角形的第N行 15.8 螺旋形矩阵 15.8.1 螺旋打印矩阵 15.8.2 生成螺旋矩阵 15.9 Z字形输出字符串 15.10 不使用乘、除、取模实现两个整数相除 15.11 文本对齐 15.12 共线的最大点数 16 其他问 16.1 随机数生成器

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值