题目链接

最长公共子序列

解题思路

第一思路:

1.用\(length[i][j]\)表示\(a\)串的前\(i\)个字符与\(b\)串的前\(j\)个字符重叠的最长子串长度

2.用\(num[i][j]\)表示 \(a\)串的前\(i\)个字符与\(b\)串的前\(j\)个字符重叠的最长子串个数

则求\(length[i][j],num[i][j]\)时有以下递推关系:

*\(length[i][j]:\)

如果当前两串结尾字符相等,则\(length[i][j]=length[i-1][j-1]+1\)

否则\(length[i][j]=max(length[i-1][j],length[i][j-1])\)

*\(num[i][j]:\)

如果\(length[i][j]\)与\(length[i-1][j]\)相等,\(num[i][j]\)可加\(num[i-1][j]\)

如果\(length[i][j]\)与\(length[i][j-1]\)相等,\(num[i][j]\)可加\(num[i][j-1]\)

如果\(length[i][j]\)与\(length[i-1][j-1]\)相等,则\(num[i][j]\)多加了个\(num[i-1][j-1]\),需要减去

代码:

#include<stdio.h>
#include<string.h>
#define max(a,b) (a>b?a:b)
int length[5010][5010],num[5010][5010],w=100000000;
char a[5010],b[5010];
int main(){
int i,j,la,lb;
scanf("%s%s",a,b);
la=strlen(a)-1;lb=strlen(b)-1;
for(i=0;i<=la;i++)num[i][0]=1;
for(i=0;i<=lb;i++)num[0][i]=1;
for(i=1;i<=la;i++){
for(j=1;j<=lb;j++){
if(a[i-1]==b[j-1]){
length[i][j]=length[i-1][j-1]+1;
num[i][j]=num[i-1][j-1];
}
else{
length[i][j]=max(length[i-1][j],length[i][j-1]);
if(length[i][j]==length[i-1][j-1])num[i][j]-=num[i-1][j-1];
num[i][j]+=w;
}
if(length[i-1][j]==length[i][j])num[i][j]+=num[i-1][j];
if(length[i][j-1]==length[i][j])num[i][j]+=num[i][j-1];
num[i][j]%=w;
length[i][j]%=w;
}
}
printf("%d\n%d",length[la][lb],num[la][lb]);
return 0;
}

提交效果

优化代码

考虑优化代码。

考虑到\(length\)、\(num\)数组当前状态都只与上一状态相关,可以用滚动数组优化空间和时间。

AC代码

#include<stdio.h>
#include<string.h>
#define max(a,b) (a>b?a:b)
int length[2][5010],num[2][5010],w=100000000;
char a[5010],b[5010];
int main(){
int i,j,la,lb;
scanf("%s%s",a,b);
la=strlen(a)-1;lb=strlen(b)-1;
for(i=0;i<=lb;i++)num[0][i]=1;
num[1][0]=1;
for(i=1;i<=la;i++){
int temp=i%2;
for(j=1;j<=lb;j++){
num[temp][j]=0;//滚动数组一定要注意这一点
if(a[i-1]==b[j-1]){
length[temp][j]=length[temp^1][j-1]+1;
num[temp][j]=num[temp^1][j-1];
}
else{
length[temp][j]=max(length[temp^1][j],length[temp][j-1]);
if(length[temp][j]==length[temp^1][j-1])num[temp][j]-=num[temp^1][j-1];
num[temp][j]+=w;
}
if(length[temp^1][j]==length[temp][j])num[temp][j]+=num[temp^1][j];
if(length[temp][j-1]==length[temp][j])num[temp][j]+=num[temp][j-1];
num[temp][j]%=w;
length[temp][j]%=w;
}
}
printf("%d\n%d",length[la%2][lb],num[la%2][lb]);
return 0;
}

提交效果

P2516 [HAOI2010]最长公共子序列 题解(LCS)的更多相关文章

  1. 洛谷P2516 [HAOI2010]最长公共子序列(LCS,最短路)

    洛谷题目传送门 一进来就看到一个多月前秒了此题的ysn和YCB%%% 最长公共子序列的\(O(n^2)\)的求解,Dalao们想必都很熟悉了吧!不过蒟蒻突然发现,用网格图貌似可以很轻松地理解这个东东? ...

  2. 2021.12.10 P2516 [HAOI2010]最长公共子序列(动态规划+滚动数组)

    2021.12.10 P2516 [HAOI2010]最长公共子序列(动态规划+滚动数组) https://www.luogu.com.cn/problem/P2516 题意: 给定字符串 \(S\) ...

  3. [HAOI2010]最长公共子序列(LCS+dp计数)

    字符序列的子序列是指从给定字符序列中随意地(不一定连续)去掉若干个字符(可能一个也不去掉)后所形成的字符序列.令给定的字符序列X=“x0,x1,…,xm-1”,序列Y=“y0,y1,…,yk-1”是X ...

  4. 洛谷 P2516 [HAOI2010]最长公共子序列

    题目传送门 解题思路: 第一问要求最长公共子序列,直接套模板就好了. 第二问要求数量,ans[i][j]表示第一个字符串前i个字符,第二个字符串前j个字符的最长公共子序列的数量 如果f[i][j]是由 ...

  5. luogu P2516 [HAOI2010]最长公共子序列

    传送门 首先那个\(O(n^2)\)的dp都会吧,不会自己找博客或者问别人,或是去做模板题(误) 对以下内容不理解的,强势推荐flash的博客 我们除了原来记录最长上升子序列的\(f_{i,j}\), ...

  6. 洛谷P2516 [HAOI2010]最长公共子序列

    题目描述 字符序列的子序列是指从给定字符序列中随意地(不一定连续)去掉若干个字符(可能一个也不去掉)后所形成的字符序列.令给定的字符序列X="x0,x1,-,xm-1",序列Y=& ...

  7. Luogu P2516 [HAOI2010]最长公共子序列 DP

    首先$LIS$显然:$f[i][j]=max(f[i][j-1],f[i-1][j],(a[i]==b[j])*f[i-1][j-1])$ 考虑如何转移数量: 首先,不管$a[i]$是否等于$b[j] ...

  8. P2516 [HAOI2010]最长公共子序列

    传送门 看到数据范围,显然 $n^2$ 的 $dp$... 设 $f[i][j]$ 表示 $A$ 串考虑了前 $i$ 位,$B$ 串考虑了前 $j$ 位,最优情况下的方案数 但是好像没法判断转移来的是 ...

  9. [BZOJ2423][HAOI2010]最长公共子序列

    [BZOJ2423][HAOI2010]最长公共子序列 试题描述 字符序列的子序列是指从给定字符序列中随意地(不一定连续)去掉若干个字符(可能一个也不去掉)后所形成的字符序列.令给定的字符序列X=“x ...

随机推荐

  1. kubernetes实战-交付dubbo服务到k8s集群(三)安装配置maven和java运行时环境的底包镜像

    maven 官方地址: 官方地址 下载maven,shdd7-200 # cd /opt/src # wget https://archive.apache.org/dist/maven/maven- ...

  2. 6. Connection has already been closed 数据库连接被关闭

    生产上Tomcat出现 Connection has already been closed.问题,但是在uat测试是好的! 遇见两次: 1.某个程序dao中执行逻辑异常复杂,有时候需要执行一分多钟, ...

  3. Redis五大类型及底层实现原理

    目录 简单动态字符串链表字典跳跃表整数集合压缩列表对象 对象的类型与编码字符串对象列表对象哈希对象 集合对象有序集合对象类型检查与命令多态内存回收对象共享对象的空转时长 简单动态字符串  导读 Red ...

  4. 机器学习(四):通俗理解支持向量机SVM及代码实践

    上一篇文章我们介绍了使用逻辑回归来处理分类问题,本文我们讲一个更强大的分类模型.本文依旧侧重代码实践,你会发现我们解决问题的手段越来越丰富,问题处理起来越来越简单. 支持向量机(Support Vec ...

  5. Steam 钓鱼模拟器

    Steam 钓鱼模拟器 Fishing Planet Fishing Planet 是一个独特和高度现实的在线第一人称多人钓鱼模拟器,由狂热的钓鱼爱好者钓鱼给你带来实际钓鱼充分刺激开发! 选择你的诱饵 ...

  6. Vue 3.x & v-model

    Vue 3.x & v-model https://v3.vuejs.org/guide/migration/v-model.html#overview BREAKING: When used ...

  7. Build your own React

    Build your own React https://pomb.us/build-your-own-react/ https://github.com/pomber/didact demo htt ...

  8. 2020 新型肺炎病毒疫情 & 远程办公

    2020 新型肺炎病毒疫情 & 远程办公 2020 新型肺炎病毒疫情 https://zhuanlan.zhihu.com/p/104406687 钉钉 微信 code gitlab PRD ...

  9. Masterboxan INC :个股出现疯涨,投资者需警惕股市泡沫

    随着标普500指数自去年三月以来的暴涨,引发了很多亏损企业股价飙升,同时许多场外投资者盲目跟风,加剧了个股的疯涨.对于此现象,美国万事达资产管理有限公司不得不多次发文提醒投资者:个股出现疯涨,投资者需 ...

  10. BGV作为拥抱新时代的DeFi项目,是否有能力超越YFI?

    随着今年11月DeFi蓝筹股们的集体反弹,市场变化让投资者明白,不能再死守诸如COMP和MKR的古典DeFi了,只有拥抱新时代的DeFi们才有赚钱的可能,不要和钱过不去.经过9-10月的回调,11月的 ...