一、最长公共子序列

  经典的动态规划问题,大概的陈述如下:

  给定两个序列a1,a2,a3,a4,a5,a6......和b1,b2,b3,b4,b5,b6.......,要求这样的序列使得c同时是这两个序列中的部分(不要求连续),这个就叫做公共子序列,然后最长公共子序列自然就是所有的子序列中最长的啦。

  既然是动态规划,难点肯定是在转移方程那了。首先我们用一张网上流传的图:

  我个人觉得这张图最好的阐述了这个问题的解法。下面说一下我的理解:首先我们要考虑怎么表示LCS中的各个状态,这个知道的可能觉得很简单,但是不知道的可能想很久也想不到,我就是在看到这张图才真正理解为什么要采用二维数组表示这个状态(可能不同的人理解不同)。我们用f[i][j]表示字符串a的前i个与字符串b的前j个的最长公共子串,则状态转移方程是:

现在我们通过上面的那个图来理解一下这个方程,这个图显示的是求ABCBDAB与BDCABA的LCS,首先当某一个串为零,公共子串自然也为0了,然后如果某一个值上两个串相同那么自然是加一,最后不等的时候由图可以看到他们的来源有两个防线即上方和左方,通过这个我们就可以一步一步求出最大的公共子串了。

 /*
总结:这个是经典的LCS问题,主要是要找到递归方程
状态用二维数组表示(想到这个是难点)用f[a][b],表示字符串
A中前a个与字符串B中前b个的最大公共子串
递归方程:
f[i][j]= f[i-1][j-1]+1(相等的时候)
max(f[i][j-1],f[i-1][j])(不等的时候)
*/
#include<iostream>
#include<algorithm>
#include<cstdio>
#include<queue>
#include<cstring>
#include<cmath>
using namespace std;
int main()
{
string a,b;
while(cin>>a>>b){
int dp[][]={};
int la=a.length();
int lb=b.length();
int maxx=;
for(int i=;i<la;i++){
for(int j=;j<lb;j++){
if(a[i]==b[j]){
dp[i+][j+]=dp[i][j]+;
if(maxx<dp[i+][j+])
maxx=dp[i+][j+];
}
else{
dp[i+][j+]=max(dp[i][j+],dp[i+][j]);
}
}
}
cout<<maxx<<endl;
}
return ;
}

  最后再稍微说一点最长公共子序列的用途,这个算法的一种用途就是用来判断两段文字的相似度,举一个我们最熟悉的例子,有两段程序,一段是原版,一段使用改变量名等手段做出来的那么他们的最长公共子序列一定会比较长,也就是说这个可以帮助我们找到看起来很不同的东西中的许多共同点。

二、最长递增子序列

  子序列这里就不用再介绍了,最长递增子序列也很好理解,就是给定一个序列 An a1 ,a2 ,  ... , an ,找出最长的子序列使得对所有 j ,ai aj 。。这个问题又是一个动态规划。它有两种解决方案:

  1、转化成LCS问题求解,初看这个貌似是没什么关系,但是我们至腰间这个序列排序之后与原来的求LCS得到的就是LIS了,个人觉得这个特别的    好,如果你自己能想出来那么你对LCS的理解也有一定深度了。

  2、继续动态规划,首先我们用f[i]表示以i结尾的LIS的长度,,这时我们就要考虑f[i]与前面那些值的关系,即这个值是要加载他前面那    个比他小的值上,这样才能达到最长。初始值自然是全部为1。

这里就不直接贴代码了,太简单了,给一个有一定改动的例子(其实没有本质性变化),合唱团:

题目描述:

N位同学站成一排,音乐老师要请其中的(N-K)位同学出列,使得剩下的K位同学不交换位置就能排成合唱队形。
合唱队形是指这样的一种队形:设K位同学从左到右依次编号为1, 2, …, K,他们的身高分别为T1, T2, …, TK,
则他们的身高满足T1 < T2 < … < Ti , Ti > Ti+1 > … > TK (1 <= i <= K)。
你的任务是,已知所有N位同学的身高,计算最少需要几位同学出列,可以使得剩下的同学排成合唱队形。

输入:

输入的第一行是一个整数N(2 <= N <= 100),表示同学的总数。
第一行有n个整数,用空格分隔,第i个整数Ti(130 <= Ti <= 230)是第i位同学的身高(厘米)。

输出:

可能包括多组测试数据,对于每组数据,
输出包括一行,这一行只包含一个整数,就是最少需要几位同学出列。

 /*
总结:这个是经典的LIS问题两种解决方法:
一、排序后转换成LCS
二、直接找转化方程,令L[i]表示以i结尾的最长递增子串长度,
则L[i]=max(l[j]+1,1)其中j表示前面小于i的且a[i]>a[j]的值
*/
#include<iostream>
#include<algorithm>
#include<cstdio>
#include<queue>
#include<cstring>
#include<cmath>
using namespace std;
int main()
{
int n;
while(scanf("%d",&n)==&&n){
int a[];
int b[];
int c[];
int d[];
for(int i=;i<n;i++){
b[i]=;
c[i]=;
d[i]=;
scanf("%d",&a[i]);
for(int j=i-;j>=;j--){
if(a[i]>a[j]){
b[i]=max(b[i],b[j]+);
}
}
}
for(int i=n-;i>=;i--){
for(int j=i+;j<n;j++){
if(a[i]>a[j]){
c[i]=max(c[i],c[j]+);
}
}
}
int maxx=;
for(int i=;i<n;i++){
d[i]=b[i]+c[i];
//cout<<b[i]<<ends<<c[i]<<ends<<d[i]<<endl;
}
for(int i=;i<n;i++){
if(maxx<d[i]){
maxx=d[i];
}
}
printf("%d\n",n-maxx+);
}
return ;
}

最长公共子序列(LCS)和最长递增子序列(LIS)的求解的更多相关文章

  1. 最长公共子串(LCS:Longest Common Substring)

    最长公共子串(LCS:Longest Common Substring)是一个非常经典的面试题目,本人在乐视二面中被面试官问过,惨败在该题目中. 什么是最长公共子串 最长公共子串问题的基本表述为:给定 ...

  2. 每日一题-——最长公共子序列(LCS)与最长公共子串

    最长公共子序列(LCS) 思路: 代码: def LCS(string1,string2): len1 = len(string1) len2 = len(string2) res = [[0 for ...

  3. 最长连续公共子序列(LCS)与最长递增公共子序列(LIS)

    最长公共子序列(不连续) 实际问题中也有比较多的应用,比如,论文查重这种,就是很实际的一个使用方面. 这个应该是最常见的一种了,不再赘述,直接按照转移方程来进行: 按最普通的方式就是,直接构造二维矩阵 ...

  4. 最长公共子窜和最长公共子序列(LCS)

    他们都是用dp做;复杂度都是O(N方) 有一个大佬的博客写的很详细,是关于最长公共子序列的:https://blog.csdn.net/hrn1216/article/details/51534607 ...

  5. 最长回文子序列LCS,最长递增子序列LIS及相互联系

    最长公共子序列LCS Lintcode 77. 最长公共子序列 LCS问题是求两个字符串的最长公共子序列 \[ dp[i][j] = \left\{\begin{matrix} & max(d ...

  6. 最长公共子串LCS(Longest Common Substring)

    一.问题描述 寻求两个字符串中的最大公共字串,其中子串是指字符串中连续的字符组成的,而不是像子序列,按照字符的前后顺序组成.如str1="sgabacbadfgbacst",str ...

  7. [算法练习]最长公共子串(LCS)

    题目说明: 找两个字符串的最长公共子串,这个子串要求在原字符串中是连续的.比如"bab"和"caba"的最长公共子串是"ba"和" ...

  8. 最长公共子串(LCS) lg SP1811

    后缀自动机的一大用处就是求最长公共子串了 这道题的话题意就是给你两个字符串,求最长公共子串 做法的话是先使用一个字符串建立SAM,然后让另一个串在上面进行匹配 匹配的策略是优先匹配当前节点的下一个字符 ...

  9. 求两个字符串的最长公共子串(LCS)

    http://tianyunpu2008.blog.163.com/blog/static/6559379920089162236915/

  10. 动态规划求最长公共子序列(Longest Common Subsequence, LCS)

    1. 问题描述 子串应该比较好理解,至于什么是子序列,这里给出一个例子:有两个母串 cnblogs belong 比如序列bo, bg, lg在母串cnblogs与belong中都出现过并且出现顺序与 ...

随机推荐

  1. jQuery清除文本框,内容并设置不可用

    JQuery清除文本框,内容并设置不可用  如果是设置只读,则将disabled换成readonly function CleanText(textid) { $("#"+text ...

  2. CEF3开发者系列之工程和代码结构

    CEF支持一系列的编程语言和操作系统,并且能很容易地整合到新的或已有的工程中去.它的设计思想就是易用且兼顾性能. CEF3支持一系列的编程语言和操作系统,并且能很容易地整合到新的或已有的工程中去.它的 ...

  3. ACM/ICPC 之 模拟 (HNUOJ 13391-换瓶模拟)

    题意:汽水瓶有三个部分cap+plastic bottle+ label(瓶盖-瓶身-瓶底),给出数据:n为原瓶数,x,y,z为这三个部分可以用相应的数字换取新瓶子,求最大总瓶数. 模拟(暴力) // ...

  4. 头文件algorithm中的常用函数

    非修改性序列操作(12个) 循环         对序列中的每个元素执行某操作         for_each() 查找         在序列中找出某个值的第一次出现的位置         fin ...

  5. git remote 相关用法

    为了便于管理,Git要求每个远程主机都必须指定一个主机名.git remote  命令就用于管理主机名. 不带选项的时候,git remote命令列出所有远程主机. $ git remote orig ...

  6. 【python】id()函数

    来源:百度知道 >>> a=2.0 >>> b=2.0 >>> id(a) 524440880 >>> id(b) 524440 ...

  7. [jquery]判断页面滚动到顶部和底部(适用于手机web加载)

    //判断页面滚动到顶部和底部 $(window).scroll(function(){ var doc_height = $(document).height(); var scroll_top = ...

  8. iOS应用架构谈(一):架构设计的方法论

    当我们讨论客户端应用架构的时候,我们在讨论什么? 其实市面上大部分应用不外乎就是颠过来倒过去地做以下这些事情: 简单来说就是调API,展示页面,然后跳转到别的地方再调API,再展示页面. 那这有什么好 ...

  9. [Linux] linux awk命令详解

    reference : http://www.cnblogs.com/ggjucheng/archive/2013/01/13/2858470.html 简介 awk是一个强大的文本分析工具,相对于g ...

  10. oracle sqlplus存储过程控制台输出信息

    如果你是使用PL/sql工具,在command 窗口下执行set serveroutput on 然后exec sp;可以看到了或者在sqlplus 中执行上面的代码