最长公共子序列及其引申问题

最长公共子序列是经典的动态规划问题,在很多书籍和文章中都有介绍,这里对这一经典算法进行回顾并对两个follow up questions进行总结和分析。

1. 回顾LCS(longest common subsequence)解法,求LCS长度

典型的双序列动态规划问题,dp[i][j]表示第一个序列前i项与第二个序列的前j项....

所以对应此题,dp[i][j]表示序列s1的前i项与序列s2的前j项的最长公共子序列。

得到如下递推关系式:

dp[i][j] = dp[i - 1][j - 1] + 1                  if s1[i - 1] == s2[j - 1]

    = max(dp[i - 1][j], dp[i][j - 1])   if s1[i - 1] != s2[j - 1] ;

对dp[0][j] j = 0,1,... 于 dp[i][0] i = 0,1...初始化,根据递推关系式则可以得到结果。

程序:

 1 int longsetCommonSubsequence(const string& s1, const string& s2) {
 2     int sz1 = s1.size(), sz2 = s2.size();
 3     int dp[sz1 + 1][sz2 + 1];
 4     for (int i = 0; i <= sz1; ++i) {
 5         dp[i][0] = 0;
 6     }
 7     for (int i = 0; i <= sz2; ++i) {
 8         dp[0][i] = 0;
 9     }
10     for (int i = 1; i <= sz1; ++i) {
11         for (int j = 1; j <= sz2; ++j) {
12             if (s1[i] == s2[j]) {
13                 dp[i][j] = dp[i - 1][j - 1] + 1;
14             }
15             else {
16                 dp[i][j] = max(dp[i - 1][j], dp[i][j - 1]);
17             }
18         }
19     }
20     return dp[sz1][sz2];
21 }

 

2. 如何得到最长公共子序列?

其实本质上,当有了dp数组时,即得到了行程LCS的所有信息。

我们考察一个例子,求s1 = "ABCBDAB" 和 s2 = "BDCABA"的 LCS, 得到其dp数组如下。(图片来自邹博)

我们可以看到,当dp数组被填充完毕后。反向探索LCS的形成过程,结合递推公式可知,每一个添加到LCS中的字符是在s1[i - 1] == s2[j - 1]的情况下加入的。

也就是上述图中向同时向左上方的那些步骤,而对应s1[i - 1] != s2[j - 1]的那些步骤,则选择其dp值大的(原因在于生成的时候路径就是选的max)进行前进即可。

所以总结寻找LCS的算法步骤即:

如果s1[i - 1] == s2[j - 1],将其push_back到结果中;

如果s1[i - 1] != s2[j - 1],选择dp[i - 1][j]与dp[i][j - 1]中大的更新i--或j--。

直到i或者j达到0后,翻转上述结果即可。

代码:

 1 string longsetCommonSubsequence2(const string& s1, const string& s2) {
 2     int sz1 = s1.size(), sz2 = s2.size();
 3     int dp[sz1 + 1][sz2 + 1];
 4     for (int i = 0; i <= sz1; ++i) {
 5         dp[i][0] = 0;
 6     }
 7     for (int i = 0; i <= sz2; ++i) {
 8         dp[0][i] = 0;
 9     }
10     for (int i = 1; i <= sz1; ++i) {
11         for (int j = 1; j <= sz2; ++j) {
12             if (s1[i - 1] == s2[j - 1]) {
13                 dp[i][j] = dp[i - 1][j - 1] + 1;
14             }
15             else {
16                 dp[i][j] = max(dp[i - 1][j], dp[i][j - 1]);
17             }
18         }
19     }
20     string result;
21     int i = sz1, j = sz2;
22     while (i > 0 && j > 0) {
23         if (s1[i - 1] == s2[j - 1]) { //对应的是s1[i - 1], s2[j - 1]
24             result.push_back(s1[i - 1]);
25             i--;
26             j--;
27         }
28         else {
29             if (dp[i - 1][j] > dp[i][j - 1]) {
30                 i--;
31             }
32             else {
33                 j--;
34             }
35         }
36     }
37     reverse(result.begin(), result.end());
38     return result;
39 }

 

3. 有多组解都要输出怎么办?

首先考虑什么时候会有多组解?还考虑上述图示可以发现,当s1[i - 1] != s2[j -1]但 dp[i - 1][j] == dp[i][j - 1]时,i--, j--均可。所以可能产生多组解。

想要输出所有解的想法其实也很直观,就是DFS,把所有情况都遍历一遍,并进行去重(代码中find语句),把不同路径的相同解删除掉,即可得到所有解。

一段完整的,带简单测试的程序如下:

 1 #include <iostream>
 2 #include <string>
 3 #include <vector> 
 4 #include <algorithm>
 5 using namespace std;
 6 vector<int> temp;
 7 vector<vector<int>> resultIndex;
 8 void dfs(const vector<vector<int>>& dp, const string& s1, const string& s2, int i, int j) {
 9     if (i == 0 || j == 0) {
10         if (find(resultIndex.begin(), resultIndex.end(), temp) == resultIndex.end()) {
11             resultIndex.push_back(temp);
12         }
13         return ;
14     }
15     if (s1[i - 1] == s2[j - 1]) {
16         temp.push_back(i - 1);
17         dfs(dp, s1, s2, i - 1, j - 1);
18     }
19     else {
20         if (dp[i - 1][j] > dp[i][j - 1]) {
21             dfs(dp,s1,s2,i - 1, j);
22         }
23         else if (dp[i - 1][j] < dp[i][j - 1]) {
24             dfs(dp, s1, s2, i, j - 1);
25         }
26         else {
27             vector<int> temp2 = temp;
28             dfs(dp,s1,s2,i - 1, j);
29             temp = temp2;
30             dfs(dp,s1,s2,i, j - 1);
31         }
32     }
33     return;
34 }
35 
36 vector<string> longsetCommonSubsequence2(const string& s1, const string& s2) {
37     int sz1 = s1.size(), sz2 = s2.size();
38     vector<vector<int>> dp(sz1 + 1,vector<int>(sz2 + 1));
39     for (int i = 0; i < sz1; ++i) {
40         dp[i][0] = 0;
41     }
42     for (int i = 0; i < sz2; ++i) {
43         dp[0][i] = 0;
44     }
45     for (int i = 1; i <= sz1; ++i) {
46         for (int j = 1; j <= sz2; ++j) {
47             if (s1[i - 1] == s2[j - 1]) {
48                 dp[i][j] = dp[i - 1][j - 1] + 1;
49             }
50             else {
51                 dp[i][j] = max(dp[i - 1][j], dp[i][j - 1]);
52             }
53         }
54     }
55     vector<string> result;
56     dfs(dp,s1,s2,sz1,sz2);
57     for (int i = 0; i < resultIndex.size(); ++i) {
58         string s;
59         result.push_back(s);
60         for (int j = resultIndex[i].size() - 1; j >= 0; --j) {
61             result[i].push_back(s1[resultIndex[i][j]]);
62         }
63     }
64     return result;
65 
66 }
67 
68 int main() {
69     string s1 = "ABCBDAB", s2 = "BDCABA";
70     vector<string> result = longsetCommonSubsequence2(s1,s2);
71     for (int i = 0; i < result.size(); ++i) {
72         cout << result[i] << endl;
73     }
74     return 0;
75 }

 

转载于:https://www.cnblogs.com/wangxiaobao/p/5982901.html

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

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

相关文章

怎么用java实现打字功能_怎么用JAVA编写一个打字游戏

该楼层疑似违规已被系统折叠 隐藏此楼查看此楼这个帖子是一年前发的&#xff0c;当时对Java也不是很懂&#xff0c;我看大家都对这个感兴趣&#xff0c;我把代码改了一下&#xff0c;发布出来&#xff0c;其实还有很多的地方还没有完善...&#xff0c;这个只能算一个Demo&#…

pythonfor循环列表排序_Python Day4950(for循环语句整理)

班长的图Python for循环可以遍历任何序列的项目&#xff0c;如一个列表或者一个字符串。一、Python 循环遍历列表元素1.for i in list():2.for i in enumerate(list):2.for i in range(len(list)):for i in list():for i in enumerate(list):for i in range(len(list)):二、Pyt…

Java异常处理深入理解_关于java异常处理机制的深入理解.doc

关于java异常处理机制的深入理解.doc 关于JAVA异常处理机制的深入理解1引子TRYCATCHFINALLY恐怕是大家再熟悉不过的语句了&#xff0c;而且感觉用起来也是很简单&#xff0c;逻辑上似乎也是很容易理解。不过&#xff0c;我亲自体验的“教训”告诉我&#xff0c;这个东西可不是想…

java 格式化 布尔型_这么久才知道Java中的format很强大!

Java中允许我们对指定的对象进行某种格式化&#xff0c;从而得到我们想要的格式化样式。Format首先介绍java.text包中的FormatForamt是一个抽象基类&#xff0c;其具体子类必须实现format(Object obj, StringBuffer toAppendTo, FieldPosition pos)和parseObject(String source…

【腾讯优测干货分享】从压测工具谈并发、压力、吞吐量

本文来自于腾讯bugly开发者社区&#xff0c;非经作者同意&#xff0c;请勿转载&#xff0c;原文地址&#xff1a;http://dev.qq.com/topic/580d914e07b7fc1c26a0cf7c 前言 随着部门业务的拓展&#xff0c;我们有了很多性能测试的机会&#xff0c;但在实战中&#xff0c;慢慢发现…

MySQL5.6 PERFORMANCE_SCHEMA 说明

背景&#xff1a; MySQL 5.5开始新增一个数据库&#xff1a;PERFORMANCE_SCHEMA&#xff0c;主要用于收集数据库服务器性能参数。并且库里表的存储引擎均为PERFORMANCE_SCHEMA&#xff0c;而用户是不能创建存储引擎为PERFORMANCE_SCHEMA的表。MySQL5.5默认是关闭的&#xff0c;…

04_类与对象_课程动手动脑问题以及课后实验性问题及解答集锦

Answer: 动手动脑&#xff1a; 1——以下代码为何无法通过编译&#xff1f;哪儿出错了&#xff1f; Answer: 因为类Foo的构造函数是有一个参数的&#xff0c;所以我们在new一个Foo类的对象时必须赋予一个符合条件的实参。 2—— 请运行TestStaticInitializeBlock.java示例&…

php如果能编译就完美了,centos7 完美编译PHP7 php-7.2.10.tar.gz

1.下载去官网下载。2、上传并解压tar -zxvf php-7.2.10.tar.gz3、进入文件夹cd php-7.2.104、安装相关依赖包yum install pcre pcre-devel zlib zlib-devel openssl openssl-devel gd gd-devel libjpeg libjpeg-devel libpng libpng-devel freetype freetype-devel e2fsprogs e…

2017年php还能火多久,PHP还会火吗?

据不完全数据得知&#xff0c;我国对PHP人才非常紧缺&#xff0c;大约每年有50万人左右。伴随着近几年信息化&#xff0c;智能化&#xff0c;网络化的发展&#xff0c;PHP的发展前景也是不可估量的&#xff0c;那么&#xff0c;你知道是什么影响PHP继续火热的吗?下面我们就来分…

[JZOJ P1288] [DP]矩阵取数

kaike 传送门 07年noipT3&#xff1f; 要我写我肯定放弃 嗯没错就是这么果断 据说要 高精 DP 状态&#xff1f; 举例说明&#xff0c;假设有矩阵 a1,a2,a3,a4....an b1,b2,b3,b4....bn 假设矩阵的最大得分取法为 a1*2b1*2a2*4b2*4a3*8b3*8.....an*2^nb2*2^n&#xff1b; 可以转…

Linux命令入门

// 查看日历cal // 修改密码passwd // 查看目录和文件ls -lls // 查看当前用户信息whoami // 查看当前在线用户userswho 在Linux中&#xff0c;可以使用 vi 编辑器创建一个文本文件&#xff0c;例如&#xff1a;$ vi filename上面的命令会创建文件 filename 并打开&#xff0c;…

Bug2算法的实现(RobotBASIC环境中仿真)

移动机器人智能的一个重要标志就是自主导航&#xff0c;而实现机器人自主导航有个基本要求——避障。之前简单介绍过Bug避障算法&#xff0c;但仅仅了解大致理论而不亲自动手实现一遍很难有深刻的印象&#xff0c;只能说似懂非懂。我不是天才&#xff0c;不能看几遍就理解理论中…

策略模式场景举例

容错恢复机制 容错恢复机制是应用程序开发中非常常见的功能。那么什么是容错恢复呢&#xff1f;简单点说就是&#xff1a;程序运行的时候&#xff0c;正常情况下应该按照某种方式来做&#xff0c;如果按照某种方式来做发生错误的话&#xff0c;系统并不会崩溃&#xff0…

php写抢票脚本,火车票抢票python代码公开揭秘!

市场上很多火车票抢票软件大家应该非常熟悉&#xff0c;但很少有人研究具体是怎么实现的&#xff0c;所以觉得很神秘&#xff0c;其实很简单。下面使用Python模拟抢票程序&#xff0c;给大家揭秘抢票到底是怎么回事。该代码仅供参考&#xff0c;主要用于大家沟通交流&#xff0…

代理模式——HeadFirst设计模式学习笔记

代理模式&#xff1a;为另一个对象提供一个替身或占位符控制这个对象的访问 特点&#xff1a; 让代理对象控制对象的访问&#xff0c;被代理对象可以是远程对象&#xff08;远程代理&#xff09;&#xff0c;创建开销较大对象&#xff08;虚拟代理&#xff09;&#xff0c;或需…

cursor用法java,Cursor的基本使用方法

Cursor的基本使用方法今天在用到Cursor的时候发现&#xff0c;有很多游标相关的知识还是有欠缺&#xff0c;在网上搜了篇基础讲解的文&#xff0c;觉得还不错&#xff0c;自己整理了一下发上来。虽然很基础&#xff0c;但是有一些内容之前确实没有很扎实得掌握&#xff0c;所以…

win7(64位)php5.5-Apache2.4-mysql5.6环境安装

win7&#xff08;64位&#xff09;安装搭建 php-5.5.10 apache2.4.7 mysql-5.6.16 环境 工具/原料 php-5.5.10-Win32-VC11-x64.zip 下载地址: http://windows.php.net/download/ httpd-2.4.7-win64-VC11.zip 下载地址: http://www.apachelounge.com/download/ mysql-5.6.16-win…

php开发环境 ubuntu,Ubuntu配置PHP开发环境

开发环境安装目前web服务器有很多&#xff0c;本文安装Apache服务器&#xff1b;本文使用的服务器是Mysql服务器。sudo apt install apache2常用命令重启Apache&#xff1a;sudo /etc/init.d/apache2 restart重启php&#xff1a;sudo /etc/init.d/php-fapm restart配置apache服…

php基础知识总结大全,php基础知识回顾 —— 常量

您现在的位置是&#xff1a;网站首页>>PHP>>phpphp基础知识回顾 —— 常量发布时间&#xff1a;2019-01-23 17:23:08作者&#xff1a;wangjian浏览量&#xff1a;489点赞量&#xff1a;0在PHP中有这样一类变量&#xff0c;当变量值被定义之后&#xff0c;它的值就不…

Android Studio 下使用git -- 个人,本地版本控制

第一步&#xff1a;下载安装git 下载地址 : https://git-scm.com/downloads 第二步&#xff1a;Android Studio 下配置git路径。 配置之后&#xff0c;Test弹出如下成功的提示即可。 第三步&#xff1a;创建版本库 选择项目的根目录比较适合。 第四步&#xff1a;添加需要版本控…