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

news2025/1/16 7:49:17

文章目录

  • 一、1143. 最长公共子序列
  • 二、求最长公共子序列
  • 三、变式
    • 一、1035. 不相交的线
    • 二、1312. 让字符串成为回文串的最少插入次数

一、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;
    }
};

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2067966.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

机器人开发--Pure Pursuit纯追踪介绍

机器人开发--Pure Pursuit纯追踪介绍 1 介绍1.1 概述1.2 发展历史1.3 EKF vs MPC vs Pure Pursuit1.4 PP 前探距离的影响 2 理解普渡大学--control-algorithms/basic-pure-pursuit准备导入必要的库什么是 Pure Pursuit 控制器&#xff1f;限制如何工作线圆交点线圆交点与边界选…

JUC-Synchronized原理进阶

轻量级锁 轻量级锁的使用场景&#xff1a;如果一个对象虽然有多线程要加锁&#xff0c;但加锁的时间是错开的&#xff08;也就是没有竞争&#xff09;&#xff0c;那么可以使用轻量级锁来优化。轻量级锁对使用者是透明的&#xff0c;即语法仍然是 synchronized 假设有两个方法同…

机器学习:opencv图像识别--图片专项

目录 前言 一、读取图片 1.安装opencv库 2.读取彩色图片 3.读取灰度图 二、RGB 1.RGB的概念 2.颜色通道&#xff1a; 3.图像表示 4.代码实现单通道图像 三、ROI 1.代码实现 四、图片打码 五、图片组合 六、图片缩放 总结 前言 OpenCV&#xff08;Open Source C…

Linux:Linux多线程

目录 线程概念 什么是线程 二级页表 线程的优点 线程的缺点 线程异常 线程用途 Linux进程VS线程 进程和线程 进程的多个线程共享 进程和线程的关系 Linux线程控制 POSIX线程库 线程创建 线程等待 线程终止 分离线程 线程ID及进程地址空间布局 线程概念 什么…

【CAN总线测试】——CAN数据链路层测试

从0开始学习CANoe使用 从0开始学习车载测试 相信时间的力量 星光不负赶路者&#xff0c;时光不负有心人。 目录 2.1.位时间 2.2.采样点测试 2.3.CAN报文ID和DLC一致性检查 2.4.预期帧接收测试 2.5.非预期帧接收测试 2.6.总线负载率 1.位时间 用例编号 TG2_TC1 测试目…

android aar适配uniapp

最近有商户需要接入我们sdk&#xff0c;但是我们都是android或者ios原生的&#xff0c;直接用又不能用&#xff0c;需要做适配&#xff0c;本文就教你一步步实现android aar适配uniapp。 官方参考教程&#xff1a;开发者须知 | uni小程序SDK 但是官方写的比较繁琐&#xff0c;好…

计算机毕业设计选题推荐-Cosplay论坛系统-Java/Python项目实战

✨作者主页&#xff1a;IT毕设梦工厂✨ 个人简介&#xff1a;曾从事计算机专业培训教学&#xff0c;擅长Java、Python、微信小程序、Golang、安卓Android等项目实战。接项目定制开发、代码讲解、答辩教学、文档编写、降重等。 ☑文末获取源码☑ 精彩专栏推荐⬇⬇⬇ Java项目 Py…

代码随想录训练营day27|455.分发饼干,376.摆动序列,53. 最大子序和

分发饼干 题目 思路&#xff1a;把最大的饼干分给胃口最大的人&#xff0c;所以可以先对两个数组进行排序&#xff0c;然后用双指针从后往前依次比较。如果饼干能成功头尾&#xff0c;就让饼干组的指针往前移 int biscs.size()-1; int ig.size()-1;//小孩组 for(;i>0;i--…

【摆脱被360安全卫士荼毒:使用这2个软件就够了】

保持电脑健康从拒绝使用360安全卫士开始 提示&#xff1a;不使用360安全卫士&#xff0c;电脑更健康 游戏本被360卫士荼毒的差点报废&#xff0c;感觉在使用360安全卫士后&#xff0c;笔记本的散热风扇很暴躁&#xff0c;笔记本经常高温不退&#xff0c;若你也有这样的烦恼&am…

单元格里显示曲线

想要实现的效果如下&#xff1a;表格每一行都有一个曲线 TreeList与GridControl的设置方法类似。 1、先创建控件的数据源&#xff0c;我使用的是DataTable /// <summary>/// 生成一个DataTable/// </summary>/// <returns></returns>public static Da…

【c++】强制类型转化

一、前言 在C语言中新增了四个关键字static_cast、const_cast、reinterpret_cast和dynamic_cast。这四个关键字都是用于强制类型转换的。 新类型的强制转换可以提供更好的控制强制转换过程&#xff0c;允许控制各种不同种类的强制转换。 C中风格是static_cast<type>(c…

进阶SpringBoot之 SpringSecurity(2)用户认证和授权

Spring Security 官网 Spring Security 是针对 Spring 项目的安全框架 也是 Spring Boot 底层安全模块默认的技术选型 它可以实现强大的 Web 安全控制 只需引入 spring-boot-starter-security 模块&#xff0c;进行少量配置&#xff0c;即可实现强大的安全管理 几个重要的…

首款国产“3A”游戏《黑神话:悟空》爆火,背后有哪些AI技术在助力?

近日&#xff0c;沉淀了4年的首款国产“3A”游戏《黑神话悟空》正式在各大游戏交易平台上线。 据市场研究公司VG Insights 周四晚间发布预估数据&#xff0c;《黑神话&#xff1a;悟空》自发售以来&#xff0c;三天内在 Steam 平台的销量已突破 840 万份&#xff0c;好评率超 …

速腾32线激光雷达使用方法

速腾32线激光雷达 12V电源 实体机ubuntu22.04 ROS2-humble 一、软件安装 mkdir robosense_ws cd robosense_wsmkdir src && cd src/ git clone https://github.com/RoboSense-LiDAR/rslidar_msg.git git clone https://github.com/RoboSense-LiDAR/rslidar_sdk.gi…

数学基础(六)

一、分布 正态分布 二项式分布 均匀分布 卡方分布 二、核函数 核函数的目的&#xff1a; 将低维数据转换为高维数据 线性核函数&#xff1a; Linear核函数对数据不做任何变换 当特征已经比较丰富了&#xff0c;样本数据量巨大&#xff0c;需要进行实时得出结果时进行使用…

【GH】【EXCEL】P3: Set Conditional Formatting To Excel Data By Gh

文章目录 conditional formattingdata sourceConditional ScaleConditional Scale Conditional PercentConditional Top Percent Conditional AverageConditional Average Multiple ConditionsConditional BarConditional Bar Conditional UniqueConditional Unique Conditiona…

JUC7-共享模型之工具

线程池 自定义线程池 import lombok.extern.slf4j.Slf4j; import org.springframework.core.log.LogDelegateFactory;import java.util.ArrayDeque; import java.util.Deque; import java.util.HashSet; import java.util.concurrent.TimeUnit; import java.util.concurrent.l…

git常用操作合集

1 撤销 1.1 适用场景 如果在git上提交了commit&#xff0c;但是当前提交的代码有问题&#xff0c;需要回退到上个版本 1.2 操作命令 1、git log 查看历史提交记录及对应的commit id 找到需要回退的commit id 2、执行git reset回退到之前的状态 git reset --hard <commi…

BaseCTF [第一周]Ez Xor

笔记。 64ida打开。 走&#xff01; 逆向逆向&#xff0c;逆向往前看。 因为异或算法&#xff0c;A ^BC >>> C^BA 所以在只需要知道密钥key就可以了。 是不是头大&#xff1f; 没事 这里介绍另一种方法>>> IDA 动态调试去获取key值、密文值 。(灵活使用工…

编写程序调用元神操作系统的API

1. 背景 本文介绍了元神操作系统API的调用&#xff0c;并详细介绍了“调用元神系统API读取磁盘扇区”程序的编写以及测试结果。 2. 方法 &#xff08;1&#xff09;元神操作系统API的调用方法 元神操作系统0.4版beta4开始提供了对OS功能的调用&#xff0c;调用相关的定义如…