NYOJ_36 最长公共子序列
来源:互联网 发布:v919 ubuntu 编辑:程序博客网 时间:2024/04/29 13:12
最长公共子序列
tip:最长公共子序列也称作最长公共子串(不要求连续),英文缩写为LCS(Longest Common Subsequence)。其定义是,一个序列 S ,如果分别是两个或多个已知序列的子序列,且是所有符合此条件序列中最长的,则 S 称为已知序列的最长公共子序列。
- 输入
- 第一行给出一个整数N(0<N<100)表示待测数据组数
接下来每组数据两行,分别为待测的两组字符串。每个字符串长度不大于1000. - 输出
- 每组测试数据输出一个整数,表示最长公共子序列长度。每组结果占一行。
- 样例输入
2asdfadfsd123abcabc123abc
- 样例输出
36
看到一个很好的分析,复制过来了
求两字符序列的最长公共字符子序列
问题描述:字符序列的子序列是指从给定字符序列中随意地(不一定连续)去掉若干个字符(可能一个也不去掉)后所形成的字符序列。令给定的字符序列X=“x0,x1,…,xm-1”,序列Y=“y0,y1,…,yk-1”是X的子序列,存在X的一个严格递增下标序列<i0,i1,…,ik-1>,使得对所有的j=0,1,…,k-1,有xij=yj。例如,X=“ABCBDAB”,Y=“BCDB”是X的一个子序列。
考虑最长公共子序列问题如何分解成子问题,设A=“a0,a1,…,am-1”,B=“b0,b1,…,bm-1”,并Z=“z0,z1,…,zk-1”为它们的最长公共子序列。不难证明有以下性质:
(1) 如果am-1=bn-1,则zk-1=am-1=bn-1,且“z0,z1,…,zk-2”是“a0,a1,…,am-2”和“b0,b1,…,bn-2”的一个最长公共子序列;
(2) 如果am-1!=bn-1,则若zk-1!=am-1,蕴涵“z0,z1,…,zk-1”是“a0,a1,…,am-2”和“b0,b1,…,bn-1”的一个最长公共子序列;
(3) 如果am-1!=bn-1,则若zk-1!=bn-1,蕴涵“z0,z1,…,zk-1”是“a0,a1,…,am-1”和“b0,b1,…,bn-2”的一个最长公共子序列。
这样,在找A和B的公共子序列时,如有am-1=bn-1,则进一步解决一个子问题,找“a0,a1,…,am-2”和“b0,b1,…,bm-2”的一个最长公共子序列;如果am-1!=bn-1,则要解决两个子问题,找出“a0,a1,…,am-2”和“b0,b1,…,bn-1”的一个最长公共子序列和找出“a0,a1,…,am-1”和“b0,b1,…,bn-2”的一个最长公共子序列,再取两者中较长者作为A和B的最长公共子序列。
求解:
引进一个二维数组c[][],用c[i][j]记录X[i]与Y[j] 的LCS 的长度,b[i][j]记录c[i][j]是通过哪一个子问题的值求得的,以决定搜索的方向。
我们是自底向上进行递推计算,那么在计算c[i,j]之前,c[i-1][j-1],c[i-1][j]与c[i][j-1]均已计算出来。此时我们根据X[i] = Y[j]还是X[i] != Y[j],就可以计算出c[i][j]。
问题的递归式写成:
回溯输出最长公共子序列过程:
下面是代码:
#include<stdio.h>#include<string.h>int dp[1005][1005];int maxn(int a,int b){return a>b?a:b;}int main(){int T,i,j,la,lb;char a[1005],b[1005];scanf("%d",&T);while(T--){memset(dp,0,sizeof(dp));scanf("%s",&a); la=strlen(a);scanf("%s",&b); lb=strlen(b);for(i=1;i<=la;i++){for(j=1;j<=lb;j++){if(a[i-1]==b[j-1]){dp[i][j]=dp[i-1][j-1]+1;}else if(dp[i-1][j]>=dp[i][j-1]) {dp[i][j]=dp[i-1][j];}else dp[i][j]=dp[i][j-1];}}printf("%d\n",dp[la][lb]);}return 0;}
- NYOJ_36 最长公共子序列
- nyoj_36 最长公共子序列
- 最长公共子序列
- 最长公共子序列
- 最长公共子序列
- 最长公共子序列
- 最长公共子序列...
- 最长公共子序列
- 最长公共子序列
- 最长公共子序列
- 最长公共子序列
- 最长公共子序列
- 最长公共子序列
- 最长公共子序列
- 最长公共子序列
- 最长公共子序列
- 最长公共子序列
- 最长公共子序列
- POJ 2653 Pick-up sticks(线段规范相交)
- DOS批处理中的字符串处理详解(字符串截取)
- Visual C++ 2010开发权威指南--4 MFC对话框编程
- NYOJ-小珂的苦恼【扩展欧几里得】
- Visual C++ 2010开发权威指南--5 MFC对话框控件
- NYOJ_36 最长公共子序列
- cpp premier -- ch15(待续)
- CPP -- 获取系统时间
- CPP -- 获取可用计算机串口
- CPP -- 获取程序(*.exe)完整路径
- position的static、relative、absolute、fixed、inherit
- 基于GPU的高分一号影像正射校正的设计与实现
- Oracle SQL优化 总结
- 即使没有读者,你应该写博客(You should blog even if you have no readers)