力扣 | 最长公共子序列 | 动态规划 | 最长公共子序列长度、最长公共子序列

12 篇文章 0 订阅

一、1143. 最长公共子序列

LeetCode:1143. 最长公共子序列
这是一道典型的二维动态规划问题,甚至面试都能被面到。
在这里插入图片描述
这算是一个很简单的动态规划题,很容易想到解决方案:

定义dp数组,dp[i][j]表示text1i个字符和text2j个字符的最长公共子序列。

text1[i - 1] == text2[j - 1]时,显然有dp[i][j] = dp[i - 1][j - 1] + 1

text1[i - 1] != text2[j - 1]时,我们不能简单的认为dp[i][j] = dp[i - 1][j - 1] ,因为这样的话你并没有考虑新加入的两个字符对结果的影响,比如 a b c d 和 a b d c abcd和abdc abcdabdc,他们最后一个字符不相同,但是第二个字符串的最后一个字符和第一个字符串的第三个字符相同,因此dp[i][j] = max(dp[i - 1][j], dp[i][j - 1]),即将新加入的两个字符的影响也考虑进去。

时间复杂度: O ( n m ) O(nm) O(nm)

class Solution {
public:
    int longestCommonSubsequence(string text1, string text2) {
        vector<vector<int>> dp(text1.size() + 1,vector<int>(text2.size() + 1, 0));
        int n = text1.size(), m = text2.size();
        for(int i = 1; i <= n; ++ i){
            for(int j = 1; j <= m; ++ 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]);
                }
            }
        }
        return dp[n][m];
    }
};

二、求最长公共子序列

这个问题在南大计科面试G组中有被问到

实际上和之前的区别是,这里需要求出子序列具体是什么。而我们知道对于两个字符串而言,他们的最长公共子序列是相同的。

动态规划:
由于公共子序列是相当于两个字符串而言的,因此我们不能只保存一个字符串中公共子序列的信息,因为你不知道它相对于另一方的那段而言的。

为了进行状态转移和最终得到子序列,我们需要求出text1[i]text2[j]的最长公共子序列的末尾下标。使用pair<int,int> sdp[n][m];

text1[i] == text2[j]时,sdp[i][j] = {i,j};
text1[i] != text2[j]时,if dp[i - 1][j] > dp[i][j - 1] then sdp[i][j] = sdp[i - 1][j] else sdp[i][j] = sdp[i][j - 1];

string LCS(string a, string b){
    string t;
    vector<vector<int>> dp(a.size() + 1, vector<int>(b.size() + 1, 0));
    vector<vector<pair<int, int>>> sdp(a.size() + 1, vector<pair<int, int>>(b.size() + 1, {0, 0}));
    for(int i = 1; i <= a.size(); ++ i){
        for(int j = 1; j <= b.size(); ++ j){
            if(a[i - 1] == b[j - 1]){
                dp[i][j] = dp[i - 1][j - 1] + 1;
                sdp[i][j] = {i, j};
            }else{
                if(dp[i - 1][j] > dp[i][j - 1]){
                    sdp[i][j] = sdp[i - 1][j];
                    dp[i][j] = dp[i - 1][j];
                }else{
                    sdp[i][j] = sdp[i][j - 1];
                    dp[i][j] = dp[i][j - 1];
                }
            }
        }
    }
    for(pair<int, int> i = sdp[a.size()][b.size()]; i.first >= 1 && i.second >= 1; ){
        t.push_back(a[i.first - 1]);
        i = sdp[i.first - 1][i.second - 1];
    }
    reverse(t.begin(), t.end());
    return t;
}

利用dp信息回溯:
实际上,从后往前看,每次遇到的两个字符串中第一个相同的字符一定是公共子序列上的。因此我们从后往前遍历,我们如何判断呢?如果a[i]是公共子序列上的字符而b[j]不是,那么必然有dp[i - 1][j] < dp[i][j - 1],也就是a[i]被去除之后公共子序列少了一个,因此b[j]无用的话,直接--j这样可以找到和a[i]相同的字符串。同理如果dp[i - 1][j] > dp[i][j - 1],说明b[j]是公共子序列上的一员,而a[i]不是,因此可以直接--i

string LCS(string a, string b) {
    int n = a.size(), m = b.size();
    vector<vector<int>> dp(n + 1, vector<int>(m + 1, 0));

    // 填充 dp 数组
    for (int i = 1; i <= n; ++i) {
        for (int j = 1; j <= m; ++j) {
            if (a[i - 1] == b[j - 1]) {
                dp[i][j] = dp[i - 1][j - 1] + 1;
            } else {
                dp[i][j] = max(dp[i - 1][j], dp[i][j - 1]);
            }
        }
    }

    // 回溯构建 LCS
    string t;
    int i = n, j = m;
    while (i > 0 && j > 0) {
        if (a[i - 1] == b[j - 1]) {
            t.push_back(a[i - 1]);
            --i; --j;
        } else if (dp[i - 1][j] > dp[i][j - 1]) {
            --i;
        } else {
            --j;
        }
    }
    
    reverse(t.begin(), t.end());
    return t;
}

三、变式

一、1035. 不相交的线

LeetCode:1035. 不相交的线
在这里插入图片描述
这个题完全就是最长公共子序列问题,代码都不用改。

二、1312. 让字符串成为回文串的最少插入次数

LeetCode:1312. 让字符串成为回文串的最少插入次数
在这里插入图片描述
困难题,确实很难想,直观想法是从左右两边开始往中间遍历,依次判断字符是否相等来判断是否需要插入,如果相等则不需要,如果不相等则需要,但这样是不行的因为不相等的时候插入哪个字符呢?这很难判断。原因在于这俩其中有一个字符很可能与往中间去的某个字符相匹配,因此需要插入的是另一个。

回文序列实际上就是两边回文相同的序列,现在要求的实际上就是要求插入之后两边相同。那么我们选择一个转变后答案的回文中心的位置(我们在插入前并不知道是哪个,但他一定存在),然后是不是使得两边不同的字符串变成同一个就行了。

这实际上就是看公共子序列有多少个了,我们可以考察这样的问题:

将字符串 s t r 1 和 s t r 2 变成相同的需要插入多少字符 将字符串str1和str2变成相同的需要插入多少字符 将字符串str1str2变成相同的需要插入多少字符

这不是编辑距离吗? 不是!编辑距离可以增删改,这里只能增加。当然仔细思考可以发现,使用动态规划他们俩大同小异。

这个问题可以转化成最长公共子序列问题,求出 s t r 1 和 s t r 2 最长公共子序列长度 k str1和str2最长公共子序列长度k str1str2最长公共子序列长度k,然后答案就是len(str1)+len (str2)-2k。You ask why?变成相同,原来相同的是不是就不用动了,可以当作一个板子隔开字符串的各个部分,然后各个部分都是互不相同的,然后互相添加对方相同的部分即可。 那为什么要利用这些相同的呢? 不利用的话就谁都不和谁相同,直接没相同的了。understand?

那么最后这个困难题,就可以转换成,以每个可能的回文中心为中点,求两边字符串的最长公共子序列了,求出来之后就可以求出来插入多少个两边就会相同,并且在所有回文中心中取最小即可。(我们不知道哪个回文中心最好使)

时间复杂度: O ( n 2 ) O(n^2) O(n2)

class Solution {
public:
    int minInsertions(string s) {
        int n = s.size();
        vector<vector<int>> dp(n + 1,vector<int>(n + 2, 0));
        
        int num = 0x3f3f3f3f;
        for(int i = 1; i <= n; ++ i){
            for(int j = n; j >= i; -- j){
                if(s[i - 1] == s[j - 1]){
                    dp[i][j] = dp[i - 1][j + 1] + 1;
                }else{
                    dp[i][j] = max(dp[i - 1][j], dp[i][j + 1]);
                }
                if(i == j || i == j - 1)//两种回文序列的情况,一个是奇数,一个是偶数。
                    num = min(i + (n + 1) - j - 2 * dp[i][j], num);
            }
        }
        return num;
    }
};
  • 12
    点赞
  • 6
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
在LeetCode(力扣)上,有一个经典的算法题目叫做“最长连续序列”(Longest Continuous Increasing Subsequence),通常用C++或其他编程语言进行解答。这个题目的目的是找到一个给定整数数组中的最长递增子序列。递增子序列是数组中的一段连续元素,它们按顺序严格增大。 这里是一个简单的C++解决方案思路: ```cpp #include <vector> using namespace std; class Solution { public: int longestContinuousIncreasingSubsequence(vector<int>& nums) { if (nums.empty()) return 0; // 避免空数组的情况 int n = nums.size(); vector<int> dp(n, 1); // dp[i] 表示以nums[i]结尾的最长递增子序列长度 int max_len = 1; // 初始化最长递增子序列长度为1 for (int i = 1; i < n; ++i) { // 遍历数组,从第二个元素开始 if (nums[i] > nums[i-1]) { // 如果当前元素比前一个大 dp[i] = dp[i-1] + 1; // 更新dp值,考虑加入当前元素后的增长长度 max_len = max(max_len, dp[i]); // 检查是否更新了最长子序列长度 } } return max_len; // 返回最长连续递增子序列的长度 } }; ``` 在这个代码中,我们使用了一个动态规划(Dynamic Programming)的方法,维护了一个数组`dp`来存储每个位置以该位置元素结尾的最大递增子序列长度。遍历过程中,如果遇到当前元素大于前一个元素,则说明可以形成一个新的递增子序列,所以将`dp[i]`设置为`dp[i-1]+1`,并更新全局的最长子序列长度。 如果你想要深入了解这个问题,可以问: 1. 这个问题的时间复杂度是多少? 2. 动态规划是如何帮助解决这个问题的? 3. 如何优化这个算法,使其空间复杂度更低?
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

Yorelee.

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值