`

南阳理工OJ 36 最长公共子序列

 
阅读更多

连接: http://acm.nyist.net/JudgeOnline/problem.php?pid=36

 

最长公共子序列

时间限制:3000 ms  |  内存限制:65535 KB
难度:3
 
描述
咱们就不拐弯抹角了,如题,需要你做的就是写一个程序,得出最长公共子序列。
tip:最长公共子序列也称作最长公共子串(不要求连续),英文缩写为LCS(Longest Common Subsequence)。其定义是,一个序列 S ,如果分别是两个或多个已知序列的子序列,且是所有符合此条件序列中最长的,则 S 称为已知序列的最长公共子序列。
 
输入
第一行给出一个整数N(0<N<100)表示待测数据组数
接下来每组数据两行,分别为待测的两组字符串。每个字符串长度不大于1000.
输出
每组测试数据输出一个整数,表示最长公共子序列长度。每组结果占一行。
样例输入
2
asdf
adfsd
123abc
abc123abc
样例输出
3
6

 

 

动态规划:p[i][j]表示第一个串的第i个和第二个串的j比较时的最优解

如果  str1[i]==str2[j]   则p[i][j]=p[i-1][j-1]+1;

否则  p[i][j]=max( p[i-1][j]  ,  p[i][j-1] );

然后  p[len1][len2]就是要求的结果啦!

 

#include<string.h>
#include<stdio.h>
#define max(a,b) a>b?a:b
int p[1010][1010];
int main()
{
       int T,i,j,len1,len2;
       char str1[1010];
       char str2[1010];
       scanf("%d",&T);
       while(T--)
       {
              memset(p,0,sizeof(p));
              scanf("%s%s",str1,str2);
              len1=strlen(str1);
              len2=strlen(str2);
              for(i=0;i<len1;i++)
              {
                     for(j=0;j<len2;j++)
                     {
                            if(str1[i]==str2[j])p[i+1][j+1]=1+p[i][j];
                            else p[i+1][j+1]=max(p[i][j+1],p[i+1][j]);
                     }
              }
              printf("%d\n",p[len1][len2]);
       }
       return 0;
}

 

这种方法比较笨拙,因为在比较第一个串的第i个时,第二个串只需用到p[i-1][j]或者p[i][j]  (j:1~len2)这两行数,而上面却用到了len1*len2的二维数组,下面的方法是运用滚动数组来优化。可以省下大量的时间和空间(建议在可以看懂第一种的前提下在看这种方法)

 

#include <stdio.h>
#include <string.h>
#define max(a,b) (a)>(b) ? (a) : (b)

char a[1005], b[1005];
short int up[1005], dab[1005];//up存放上一行的数,dad存放新一行的数
short int t, la, lb;

int dp(void)
{
	int i, j;

	memset(up, 0, sizeof(up));
	memset(dab, 0, sizeof(dab));
	for (i = 0; i < la; i++)//两个循环还是一样
	{
		for (j = 0; j < lb; j++)
		{
			if (a[i] == b[j])//如果第一个串的第i个和第二个串的第j个相等
				dab[j+1] = up[j]+1;//就在上一行j-1的基础上加1
			else
				dab[j+1] = max(up[j+1], dab[j]);//否则就取上一行的第j个位置和这一行的第j-1的位置的最大值
		}
		for (j = 0; j <= lb; j++)//数组滚动,更新上一行的数组
			up[j] = dab[j];
	}
	return dab[lb];//返回结果
}

int main()
{
	scanf("%hd", &t);
	while (t--)
	{
		memset(a, 0, sizeof(a));
		memset(b, 0, sizeof(b));
		scanf("%s%s", a, b);
		la = strlen(a);
		lb = strlen(b);
		printf("%hd\n", dp());
	}
	return 0;
}

 

题到此也就告一段落了,但仔细想想,上面的方法其实还是可以在次优化的,每次在找p[i][j]时,用到的只有p[i-1][j-1]  、p[i][j-1] 、 p[i-1][j]  这些值,我们找个变量把它们存起来就好了,如果只用一个数组p[j]在写

p[i-1][j]和p[i][j]其实就是p[j]一个人,只不过是在更新之前当做p[i-1][j]用,更新之后就变成了p[i][j],大家慢慢体会吧、

下面这种做法又优化了大量的时间和空间

 

#include <cstdio>
#include<cstring>
#define max(x,y)  x>y?x:y
char a[1005],b[1005];
int f[1005];
int main()
{
  int T,la,lb,k,k1,i,j;
  scanf("%d",&T);
  while(T--)
  {
      memset(f,0,sizeof(f));
      scanf("%s%s",a,b);
      la=strlen(a);
      lb=strlen(b);
      for(i=0;i<la;i++)
          for(j=0,k1=0;j<lb;j++)
          {   k=f[j+1];//k保存上一行的第j个位置的值
              if(a[i]==b[j]) f[j+1]=k1+1;
              else if(f[j]>f[j+1]) f[j+1]=f[j];//f[j+1]有双重的身份,比较前是上一行的值,比较后就是新一行的值啦!慢慢理解吧
              k1=k;//k1保存上一行的个j-1个位置的值
          }
          printf("%d\n",f[lb]);
  }
}

 

 

 

 

1
9
分享到:
评论

相关推荐

    南阳理工oj离线题库

    南阳理工oj离线题库是为编程爱好者和学习者提供的一种资源,主要用于练习和提高编程技能。这个离线题库通常包含多种类型的编程题目,涵盖了数据结构、算法、计算机科学基础等多个方面。在这个环境中,用户可以不受...

    南阳理工学院OJ_个人AC代码包(Java提交)

    【南阳理工学院OJ_个人AC代码包(Java提交)】是针对Java初学者的一份宝贵资源,它包含了参与ACM国际大学生程序设计竞赛(ICPC)时在南阳理工学院在线评测系统(OJ)上获得正确答案的代码实例。这些代码展示了如何用...

    南阳理工学院OJ第1版解题报告V1.0.pdf

    ### 南阳理工学院OJ第1版解题报告概览 #### 1. A+B Problem 虽然解题思路在报告中被省略,但我们可以推测这是一个基础的数学加法问题,涉及到数字输入与基本算术操作。此类题目旨在测试初学者对编程语言基本输入...

    南阳理工oj stl练习ac代码

    南阳理工学院的OJ(Online Judge)平台为学生提供了丰富的STL练习题目,通过AC(Accepted,表示代码正确通过所有测试用例)的代码,我们可以学习到STL在实际问题解决中的应用。 1. 容器: STL包含多种容器,如...

    最长上升子序列

    这些子序列中最长的长度是4,比如子序列(1, 3, 5, 8). 你的任务,就是对于给定的序列,求出最长上升子序列的长度。 输入样例 7 1 7 3 5 9 4 8 6 1 8 3 6 5 9 5 1 2 3 4 5 0 输出样例 4 4 5 提示 一,对输入...

    湖南理工oj题解(学习用)-共230道题

    【标题】:“湖南理工oj题解(学习用)-共230道题”揭示了这是一个针对湖南理工大学在线编程竞赛平台(Online Judge,简称OJ)的题解集合,包含了230个不同题目。这类资源通常由参赛者或者经验丰富的程序员整理,...

    1142_最大公共子序列问题

    算法设计与分析 最大公共子序列问题 。

    哈理工oj 1084百步穿杨

    哈理工OJ1084答案哈理工OJ1084答案哈理工OJ1084答案哈理工OJ1084答案哈理工OJ1084答案

    湖南理工学院OJ-小鱼比可爱

    湖南理工学院小鱼比可爱OJ题

    最长公共子系列的算法

    结课实习中做的一个小程序,是关于数据结构中最长公共子系列的算法 能运行……

    oj刷题 西安理工大学学生在线实验系统编程题答案(超级详细)

    西安理工大学的在线实验系统编程题答案集合是一份非常宝贵的资源,尤其对于正在学习编程和准备在线编程竞赛(Online Judge,简称OJ)的学生而言。这个压缩包文件包含了各种编程题目及其详细解答,可以帮助学习者深入...

    基于Laravel 5.0的OJ题解网站 , 目前涵盖安科OJ,南阳OJ,杭电OJ ,北大OJ,浙大OJ.zip

    【描述】中提到的“目前涵盖安科OJ,南阳OJ,杭电OJ,北大OJ,浙大OJ”意味着这个题解网站已经集成了多个知名OJ平台的题目,用户可以在一个统一的平台上找到这些不同OJ的题目并查看解决方案。安科OJ、南阳OJ、杭电OJ...

    ACM在线评测系统 NYOJ 题库 离线看题网页版 nyoj

    NYOJ,全称为南阳理工学院在线评测系统(Nanyang Institute of Technology Online Judge),是为ACM(国际大学生程序设计竞赛)以及其他编程爱好者提供的一种在线编程练习平台。该系统支持用户提交代码并进行实时...

    OJ题目及源码

    2. 最长公共子序列:在两个序列中找到最长的公共子序列,不需连续但必须保持顺序。动态规划的方法是建立一个二维数组,记录两个序列中每个位置的最长公共子序列长度。 3. 凸多边形最优三角剖分:寻找一个多边形的...

    趣味题:柱状图排序 西安理工大学学生在线实验系统 oj

    趣味题:柱状图排序 西安理工大学学生在线实验系统 oj

    山东理工大学2016级OJ题1832

    【知识点详解】 1. **C 语言基础**:在这些题目中,主要使用了 C 语言作为编程语言,包括变量声明、输入输出、循环结构、函数定义与调用等基本概念。例如,`scanf` 用于从标准输入读取数据,`printf` 用于输出结果...

    竞赛题集南阳OJ部分习题及解答其他oj试题及解答

    竞赛题集南阳OJ部分习题及解答其他oj试题及解答提取方式是百度网盘分享地址

    uvaoj 习题题目

    3. **字符串处理**:UVa中的题目经常需要处理字符串,涉及到模式匹配(KMP、Boyer-Moore算法)、字符串查找与替换、最长公共子序列、后缀数组、AC自动机等技术。 4. **数学应用**:许多题目需要利用数学知识,如数...

    OJ_在字符串中找出连续最长的数字串

    它要求我们从一个给定的字符串中找到最长的一段连续的数字序列。这个问题涉及到字符串遍历、字符判断以及动态规划或滑动窗口等算法技巧。 首先,我们要明确问题的关键在于如何识别和提取数字串。在C语言中,我们...

    湖南理工学院OJ-阶乘求和-定义函数

    湖南理工学院OJ-阶乘求和-定义函数

Global site tag (gtag.js) - Google Analytics