最长公共子序列也是动态规划中的一个经典问题。

有两个字符串 S1 和 S2,求一个最长公共子串,即求字符串 S3,它同时为 S1 和 S2 的子串,且要求它的长度最长,并确定这个长度。这个问题被我们称为最长公共子序列问题。

与求最长递增子序列一样,我们首先将原问题分割成一些子问题,我们用 dp[i][j]表示 S1 中前 i 个字符与 S2 中前 j 个字符分别组成的两个前缀字符串的最长公共子串长度。

显然的,当 i、 j 较小时我们可以直接得出答案,如 dp[0][j]必等于 0。那么,假设我们已经求得 dp[i][j](0<=i<x,0<=j<y)的所有值,考虑如何由这些值继而推得 dp[x][y],求得 S1 前 x 个字符组成的前缀子串和 S2 前 y 个字符 组成的前缀子串的最长公共子序列长度。若 S1[x] = =S2[y],即 S1 中的第 x 个字 符和 S2 中的第 y 个字符相同,同时由于他们都是各自前缀子串的最后一个字符, 那么必存在一个最长公共子串以 S1[x]或 S2[y]结尾,其它部分等价于 S1 中前 x-1 个字符和 S2中前 y-1个字符的最长公共子串。所以这个子串的长度比 dp[x-1][y-1] 又增加 1,即 dp[x][y] = dp[x-1][y-1] + 1。

相反的,若 S1[x] != S2[y],此时其最长 公共子串长度为 S1 中前 x-1 个字符和 S2 中前 y 个字符的最长公共子串长度与 S1 中前 x 个字符和 S2 中前 y-1 个字符的最长公共子串长度的较大者,即在两种 情况下得到的最长公共子串都不会因为其中一个字符串又增加了一个字符长度 发生改变。综上所述, dp[x][y] = max{dp[x-1][y],dp[x][y-1]}。

最长公共子序列的递推条件

假设有两个字符串S1和S2,其中S1的长度为n,S2的长度为m,用dp[i][j]表示S1前i个字符组成的前缀子串与S2前j个字符组成的前缀子串的最长公共子串长度,如下:

dp[0][j] = dp[i][0] = 0,其中j>=0 && j<=m,i>=0 && i<=n;

dp[i][j] = dp[i-1][j-1]+1;(S1[i] = S2[j])

dp[i][j] = max{dp[i-1][j],dp[i][j-1]};(S1[i] != S2[j])

最后可以求得dp[n][m]中保存的值即为两个原始字符串的最长公共子序列长度。

按照上面的公式可以写出最长公共子序列的算法

  1. #include "stdafx.h"
  2. #include <iostream>
  3. #include <string>
  4. using namespace std;
  5.  
  6. #define MAXSIZE 101
  7. char str1[MAXSIZE];
  8. char str2[MAXSIZE];
  9. //'l'表示dp[i][j] = dp[i][j] = dp[i - 1][j];
  10. //‘q’表示dp[i][j] = dp[i][j] = dp[i - 1][j];
  11. //'u'表示dp[i][j] = dp[i][j - 1];
  12. char path[MAXSIZE][MAXSIZE];
  13. int dp[MAXSIZE][MAXSIZE];
  14.  
  15. void printLCS(int i, int j)
  16. {
  17. if (i == || j == )
  18. return;
  19. if (path[i][j] == 'q')
  20. {
  21. printLCS(i - , j - );
  22. cout << str1[i-] << ' ';
  23. }
  24. else if (path[i][j] == 'u')
  25. printLCS(i - , j);
  26. else
  27. printLCS(i, j - );
  28.  
  29. }
  30.  
  31. int main()
  32. {
  33. int n, m;
  34. cin >> str1 >> str2;
  35. n = strlen(str1);
  36. m = strlen(str2);
  37. //初始化
  38. for (int i = ; i < n;i++)
  39. for (int j = ; j < m; j++)
  40. dp[i][j] = ;
  41. for (int i = ; i <= n; i++)
  42. for (int j = ; j <= m; j++)
  43. {
  44. if (str1[i - ] == str2[j - ])
  45. {
  46. dp[i][j] = dp[i - ][j - ] + ;
  47. path[i][j] = 'q';
  48. }
  49.  
  50. else
  51. {
  52. if (dp[i - ][j] >= dp[i][j - ])
  53. {
  54. dp[i][j] = dp[i - ][j];
  55. path[i][j] = 'u';
  56. }
  57.  
  58. else
  59. {
  60. dp[i][j] = dp[i][j - ];
  61. path[i][j] = 'l';
  62. }
  63.  
  64. }
  65. }
  66.  
  67. cout << dp[n][m] << endl;
  68. printLCS(n, m);
  69. return ;
  70. }

测试实例:

str1 = “abcbdab”;str2 = "bdcaba"。

输出如下:

虽然str1和str2的最长公共子序列有多个,根据dp[n][m]进行递归输出,只输出一个最长公共子序列。

动态规划 - 最长公共子序列(LCS)的更多相关文章

  1. 动态规划 最长公共子序列 LCS,最长单独递增子序列,最长公共子串

    LCS:给出两个序列S1和S2,求出的这两个序列的最大公共部分S3就是就是S1和S2的最长公共子序列了.公共部分 必须是以相同的顺序出现,但是不必要是连续的. 选出最长公共子序列.对于长度为n的序列, ...

  2. 动态规划----最长公共子序列(LCS)问题

    题目: 求解两个字符串的最长公共子序列.如 AB34C 和 A1BC2   则最长公共子序列为 ABC. 思路分析:可以用dfs深搜,这里使用到了前面没有见到过的双重循环递归.也可以使用动态规划,在建 ...

  3. 动态规划——最长公共子序列LCS及模板

    摘自 https://www.cnblogs.com/hapjin/p/5572483.html 这位大佬写的对理解DP也很有帮助,我就直接摘抄过来了,代码部分来自我做过的题 一,问题描述 给定两个字 ...

  4. 动态规划之最长公共子序列LCS(Longest Common Subsequence)

    一.问题描述 由于最长公共子序列LCS是一个比较经典的问题,主要是采用动态规划(DP)算法去实现,理论方面的讲述也非常详尽,本文重点是程序的实现部分,所以理论方面的解释主要看这篇博客:http://b ...

  5. 《算法导论》读书笔记之动态规划—最长公共子序列 & 最长公共子串(LCS)

    From:http://my.oschina.net/leejun2005/blog/117167 1.先科普下最长公共子序列 & 最长公共子串的区别: 找两个字符串的最长公共子串,这个子串要 ...

  6. 编程算法 - 最长公共子序列(LCS) 代码(C)

    最长公共子序列(LCS) 代码(C) 本文地址: http://blog.csdn.net/caroline_wendy 题目: 给定两个字符串s,t, 求出这两个字符串最长的公共子序列的长度. 字符 ...

  7. C++版 - Lintcode 77-Longest Common Subsequence最长公共子序列(LCS) - 题解

    版权声明:本文为博主Bravo Yeung(知乎UserName同名)的原创文章,欲转载请先私信获博主允许,转载时请附上网址 http://blog.csdn.net/lzuacm. C++版 - L ...

  8. 1006 最长公共子序列Lcs

    1006 最长公共子序列Lcs 基准时间限制:1 秒 空间限制:131072 KB 给出两个字符串A B,求A与B的最长公共子序列(子序列不要求是连续的). 比如两个串为: abcicba abdks ...

  9. POJ 1458 Common Subsequence(最长公共子序列LCS)

    POJ1458 Common Subsequence(最长公共子序列LCS) http://poj.org/problem?id=1458 题意: 给你两个字符串, 要你求出两个字符串的最长公共子序列 ...

  10. 51Nod 1006:最长公共子序列Lcs(打印LCS)

    1006 最长公共子序列Lcs  基准时间限制:1 秒 空间限制:131072 KB 分值: 0 难度:基础题  收藏  关注 给出两个字符串A B,求A与B的最长公共子序列(子序列不要求是连续的). ...

随机推荐

  1. zookeeper初识之原理

    ZooKeeper 是一个分布式的,开放源码的分布式应用程序协调服务,它包含一个简单的原语集,分布式应用程序可以基于它实现同步服务,配置维护和命名服务等. Zookeeper是hadoop的一个子项目 ...

  2. 搭建Tomcat6源代码阅读环境

    目标:使用MyEclipse8.5阅读Tomcat6源码. 第一步:在MyEclipse8.5中集成SVN插件. 第二步:从地址http://svn.apache.org/repos/asf/tomc ...

  3. php-eclipse乱码处理

    方法一:1)设置"eclipse目录下eclipse.ini文件"2)在文件结尾添加"-Dfile.encoding=UTF-8".3)重新启动eclipse, ...

  4. Maven安装本地jar

    应用场景: 有时候一些jar包(比如oracle 的 ojdbc.jar)由于种种原因,比如版权等,导致maven中央库没有该jar文件,但是却有该jar的pom文件. 这个时候,如果私服也没这jar ...

  5. inline-block

    在CSS中,块级对象元素会单独占一行显示,多个block元素会各自新起一行,并且可以设置width,height属性:而内联对象元素前后不会产生换行,一系列inline元素都在一行内显示,直到该行排满 ...

  6. phalcon:跟踪sql语句

    在phalcon里有一个\Phalcon\Db\Profiler 类,这个类可以用来记录sql语句并计算消耗的时间.那么如何使用它呢? 手册里其实已经提供了方法,总结如下: 1.向$di里注册prof ...

  7. C#与XML Schema的问题

    http://bbs.csdn.net/topics/50493564 weileily: 用XmlSchema.Read方法读取了一个xsd文件,请问如何遍历检索器中的ComplexType与Sim ...

  8. Flex Builder读书笔记(二)——MXML

    MXML类似于HTML,它提供各种标签来定义用户界面,但是它比HTML的结构更为严格,并拥有跟多的应用标签.MXML不仅包括可视标签,还包括不可视标签,如web service连接.数据绑定和动画效果 ...

  9. Android——使用SQLiteDatabase操作SQLite数据库

    除了可以使用文件或SharedPreferences存储数据,还可以选择使用SQLite数据库存储数据. 在Android平台上,集成了一个嵌入式关系型数据库-SQLite,SQLite3支持 NUL ...

  10. windows github 命令行使用

    1.下载git客户端工具,以下2个网站一样的https://git-for-windows.github.io/https://git-scm.com/download/win/ 2.初始化ssh-k ...