一、最长公共子序列

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

  给定两个序列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. algorithm 中的常用函数

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

  2. 带中文的路径导致NSURL初始化一直为null的问题

    一.问题描述 在学习Ojective-C过程中,需要读取文件中的内容,但发现指针变量url的值一直为nil. 代码如下: NSString *strUrl=@"file:///Users/f ...

  3. [Android进阶]学习AccessibilityService实现微信抢红包插件

    在你的手机更多设置或者高级设置中,我们会发现有个无障碍的功能,很多人不知道这个功能具体是干嘛的,其实这个功能是为了增强用户界面以帮助残障人士,或者可能暂时无法与设备充分交互的人们 它的具体实现是通过A ...

  4. OKhttp基本使用介绍

    MainActivity.class package com.example.administrator.okhttp3; import android.support.v7.app.AppCompa ...

  5. 解决【无法启动此程序,因为计算机中丢失MSVCP71.dll。尝试重新安装该程序以解决此问题】问题

    今日电脑偶然出现:[无法启动此程序,因为计算机中丢失MSVCP71.dll.尝试重新安装该程序以解决此问题.]的不断地弹窗报错. 在网上查找了好久,也没有找到解决办法,就在我将要准备重装系统的时候,又 ...

  6. tableView滚到最后一行

    dispatch_async(dispatch_get_main_queue(), ^{ [_tableview scrollToRowAtIndexPath:[NSIndexPath indexPa ...

  7. [Linux] vim的高亮查找操作

    reference :  http://blog.chinaunix.net/uid-20732478-id-763411.html 使用了VIM这么久,却一直无法牢记一些基本的操作指令.今天查找一个 ...

  8. August 25th 2016 Week 35th Thursday

    Every man dies, but not every man really lives. 每个人都会死,但不是每个人都曾经真真活过. As I become older and older, I ...

  9. nohup命令

    nohup就是不挂起的意思( n ohang up). .nohup command 或者 nohup command & 这之间的差别是带&的命令行,即使terminal(终端)关闭 ...

  10. 方法重载的小demo

    方法的重载(overload)要求:1,同一个类中2,方法名必须相同3,方法的参数列表不同(1,参数的个数不同2,参数类型不同,但是参数名相同) 注:方法的重载与方法的返回值类型没有关系 packag ...