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

文章目录

  • 一、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]);}}}// 回溯构建 LCSstring 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.mzph.cn/pingmian/52155.shtml

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

相关文章

C#关于多线程的线程问题

using System.Text; ​ namespace 平时练习8._19day06 {internal class Program{static async Task Main(string[] args){Console.WriteLine(Thread.CurrentThread.ManagedThreadId );StringBuilder sb new StringBuilder();for (int i 0; i < 10000; i){sb.Append("…

坚持绿色发展的上海智算中心,稳步推进中

自今年年初正式封顶以来&#xff0c;云端股份上海智算中心在外墙及内部的建设进展顺利。这座智算中心地理位置优越&#xff0c;正逐步成为推动数字经济发展的重要力量。 位置优势 云端股份上海智算中心毗邻智慧岛数据产业园&#xff0c;是崇明区目前建设的唯一一座智算中心&am…

多功能秒达工具箱全开源源码,可自部署且完全开源的中文工具箱

简介&#xff1a; 多功能秒达开源工具箱源码&#xff0c;&#xff0c;可自部署且完全开源的中文工具箱&#xff0c;永远的自由软件&#xff0c;轻量级运行&#xff0c;全平台支持&#xff08;包括ARMv8&#xff09;&#xff0c;完全类似 GPT 的支持&#xff0c;与高效的 UI 高…

简化登录流程,助力应用建立用户体系

随着智能手机和移动应用的普及&#xff0c;用户需要在不同的应用中注册和登录账号&#xff0c;传统的账号注册和登录流程需要用户输入用户名和密码&#xff0c;这不仅繁琐而且容易造成用户流失。 华为账号服务&#xff08;Account Kit&#xff09;提供简单、快速、安全的登录功…

一文5000字从0到1使用Jmeter实现轻量级的接口自动化测试

接口测试虽然作为版本的一环&#xff0c;但是也是有一套完整的体系&#xff0c;有接口的功能测试、性能测试、安全测试&#xff1b;同时&#xff0c;由于接口的特性&#xff0c;接口的自动化低成本高收益的&#xff0c;使用一些开源工具或一些轻量级的方法&#xff0c;在测试用…

【Kotlin设计模式】Kotlin实现工厂模式

前言 工厂模式&#xff08;Factory Pattern&#xff09;是一种创建型设计模式&#xff0c;提供一个创建对象的接口&#xff0c;不暴露对象的创建过程。它将实例化对象的任务交给子类或具体实现&#xff0c;从而使得客户端代码与具体类解耦。 工厂模式主要分为以下三类&#xf…

图像处理之:Video Processing Subsystem(一)

免责声明&#xff1a; 本文所提供的信息和内容仅供参考。作者对本文内容的准确性、完整性、及时性或适用性不作任何明示或暗示的保证。在任何情况下&#xff0c;作者不对因使用本文内容而导致的任何直接或间接损失承担责任&#xff0c;包括但不限于数据丢失、业务中断或其他经济…

ucharts图表滚动

背景&#xff1a; 使用ucharts绘制折线图&#xff0c;当数据项多的时候&#xff0c;横坐标显示的文字会重合&#xff0c;故想到滑动 项目代码使用的是原生的代码&#xff0c;而非ucharts的组件&#xff1a; <template><view><canvas canvas-id"chartsLi…

各类软件历史版本的下载地址

postman,notpad等 https://www.filehorse.com/software-developer-tools/https://www.filehorse.com/software-developer-tools/

FreeRTOS学习:内存管理

FreeRTOS内存管理简介 在使用 FreeRTOS 创建任务、队列、信号量等对象的时候&#xff0c; FreeRTOS 一般都提供了两种方法&#xff0c; 动态方法创建&#xff1a;自动地从 FreeRTOS 管理的内存堆中申请所创建对象所需的内存&#xff0c;在对象被删除后&#xff0c;又可以将这…

day33(mysql57主从从+mycat读写分离+java项目结合mycat数据库+lvs_dr轮询调用java项目)

1.设置主从从mysql57服务器 &#xff08;1&#xff09;配置主数据库 [rootmsater_5 ~]# systemctl stop [rootmsater_5 ~]# setenforce 0 [rootmsater_5 ~]# systemctl disable Removed symlink /etc/systemd/system/multi-user.target.wants/# ls anaconda-ks.cfg mysql-5.7…

Qt实现圆型控件的三种方法之设置样式表

前言 最近在研究绘制各种形状的控件&#xff0c;这里专门挑出圆形的控件进行记录&#xff0c;其它形状的也大差不差&#xff0c;会了圆形的之后其它的也类似。 正文 这里我挑出Label来进行举例。 通过设置样式表 (QSS) 这种方法简单且适用于不需要自定义绘制的场景。就是要…

vue2项目 预渲染 Unable to prerender all routes 错误排查与解决方案

前言 今天在做我的Vue2项目的SEO优化时&#xff0c;我采用了prerender-spa-plugin与vue-meta-info的一套官网前端SEO预渲染方案。 但是&#xff0c;在打包时&#xff0c;报错Unable to prerender all routes。尝试了很多种网上方案&#xff0c;都没有成功&#xff0c;最后从源…

基于vue3和audio封装的简易音频播放器

样式如图所示 <template><div class"audio-player"><div class"player_top" flex-ac flex-justify-between ><div class"fileName genericTitle" fs-28 l-height-32 height-64 pr-42 flex-ac><span class"t…

维护和升级LabVIEW程序

在维护和升级LabVIEW程序时&#xff0c;需要全面考虑代码的现状和未来的需求。以下是各个方面的详细注意事项&#xff0c;以确保程序能够在稳定性和性能方面得到提升。 1. 理解现有代码: 深入代码分析: 仔细阅读现有的代码&#xff0c;了解其逻辑结构、数据流、和控制流程。关注…

中国大模型平台市场份额最新排名Top5!场景化应用昭示新蓝海

8月21日&#xff0c;IDC&#xff08;国际数据公司&#xff09;首次发布了《中国大模型平台市场份额&#xff0c;2023&#xff1a;大模型元年 —— 初局》。数据显示&#xff0c;2023年中国大模型平台及相关应用市场规模达17.65亿元人民币。 在过去的2023年&#xff0c;行业对于…

会话跟踪方案:Cookie Session Token

什么是会话技术&#xff1f; Cookie 以登录为例&#xff0c;用户在浏览器中将账号密码输入并勾选自动登录&#xff0c;浏览器发送请求&#xff0c;请求头中设置Cookie&#xff1a;userName:张三 ,password:1234aa &#xff0c;若登录成功&#xff0c;服务器将这个cookie保存…

「数组」数组双指针算法合集:二路合并|逆向合并|快慢去重|对撞指针 / LeetCode 88|26|11(C++)

目录 概述 1.二路合并 思路 复杂度 Code 2.逆向合并 思路 复杂度 Code 3.快慢去重 思路 复杂度 Code 4.对撞指针 思路 复杂度 Code 总结 概述 数组的线性枚举是我们学习编程时遇到的第一种枚举手段。但是它看起来有点愚蠢&#xff1a;只有一个索引i承担全部…

学生党蓝牙耳机哪个牌子性价比高?推荐四款内行精选百元耳机!

作为没有什么经济来源的学生党&#xff0c;唯一来钱的途径就是家里给的生活费。所以在选择一款蓝牙耳机时就很纠结&#xff0c;那么首先你得清楚学生党的购物习性&#xff0c;因为大部分学生党在买东西时因为经济能力问题&#xff0c;主要追求的还是性价比&#xff0c;可以在实…

IP-RDS-222、IP-PRZ-59-AM12、EG-TRZ-42-L、EG-TRZ-42-H比例减压阀放大器

IP-DAR-250、IP-DAR-43C-L、IP-DAR-43C-H、IP-RDS-222、IP-PRZ-59-AM12、EG-TRZ-42-L、EG-TRZ-42-H比例减压阀 EE-PRB、EE-PRD比例压力阀 EE-P2G、ET-P2S、EB-P2A、EE-P2A、ET-P2A、EE-P2H、EG-F2A、EU-F2A比例流量阀 EF-F3G、EU-F3G比例压力补偿流量阀 EQ-S4M、EG-S4M、EQ…