P2516 [HAOI2010]最长公共子序列 题解(LCS)
题目链接
解题思路
第一思路:
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)的更多相关文章
- 洛谷P2516 [HAOI2010]最长公共子序列(LCS,最短路)
洛谷题目传送门 一进来就看到一个多月前秒了此题的ysn和YCB%%% 最长公共子序列的\(O(n^2)\)的求解,Dalao们想必都很熟悉了吧!不过蒟蒻突然发现,用网格图貌似可以很轻松地理解这个东东? ...
- 2021.12.10 P2516 [HAOI2010]最长公共子序列(动态规划+滚动数组)
2021.12.10 P2516 [HAOI2010]最长公共子序列(动态规划+滚动数组) https://www.luogu.com.cn/problem/P2516 题意: 给定字符串 \(S\) ...
- [HAOI2010]最长公共子序列(LCS+dp计数)
字符序列的子序列是指从给定字符序列中随意地(不一定连续)去掉若干个字符(可能一个也不去掉)后所形成的字符序列.令给定的字符序列X=“x0,x1,…,xm-1”,序列Y=“y0,y1,…,yk-1”是X ...
- 洛谷 P2516 [HAOI2010]最长公共子序列
题目传送门 解题思路: 第一问要求最长公共子序列,直接套模板就好了. 第二问要求数量,ans[i][j]表示第一个字符串前i个字符,第二个字符串前j个字符的最长公共子序列的数量 如果f[i][j]是由 ...
- luogu P2516 [HAOI2010]最长公共子序列
传送门 首先那个\(O(n^2)\)的dp都会吧,不会自己找博客或者问别人,或是去做模板题(误) 对以下内容不理解的,强势推荐flash的博客 我们除了原来记录最长上升子序列的\(f_{i,j}\), ...
- 洛谷P2516 [HAOI2010]最长公共子序列
题目描述 字符序列的子序列是指从给定字符序列中随意地(不一定连续)去掉若干个字符(可能一个也不去掉)后所形成的字符序列.令给定的字符序列X="x0,x1,-,xm-1",序列Y=& ...
- 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] ...
- P2516 [HAOI2010]最长公共子序列
传送门 看到数据范围,显然 $n^2$ 的 $dp$... 设 $f[i][j]$ 表示 $A$ 串考虑了前 $i$ 位,$B$ 串考虑了前 $j$ 位,最优情况下的方案数 但是好像没法判断转移来的是 ...
- [BZOJ2423][HAOI2010]最长公共子序列
[BZOJ2423][HAOI2010]最长公共子序列 试题描述 字符序列的子序列是指从给定字符序列中随意地(不一定连续)去掉若干个字符(可能一个也不去掉)后所形成的字符序列.令给定的字符序列X=“x ...
随机推荐
- 新闻类爬虫库:Newspaper
newspaper库是一个主要用来提取新闻内容及分析的Python爬虫框架.此库适合抓取新闻网页.操作简单易学,即使对完全没了解过爬虫的初学者也非常的友好,简单学习就能轻易上手,除此之外,使用过程你不 ...
- 解决宝塔面板没有命令行问题 && 查看宝塔面板项目环境
# 宝塔面板没有命令行,无法查看错误输出 利用ssh.比如xshell,MObaxtern .输入ip,username,password就可以进入服务器的命令行. # 查看项目的环境 服务器默认的p ...
- 流水线cpu —Verilog HDL
一.准备工作 先看看书(<计算机原理与设计 Verilog HDL版>),搞懂一点原理.然后照着书上的代码写一写(用8.4的就可以了,不用8.6的). 注意mux2x32,mux4,cla ...
- codeforces 5C
C. Longest Regular Bracket Sequence time limit per test 2 seconds memory limit per test 256 megabyte ...
- App icons generator
App icons generator https://appicon.co/ Drag or select an app icon image (1024x1024) to generate dif ...
- 2021 从零开始打造一个自己的 UI 组件库
2021 从零开始打造一个自己的 UI 组件库 refs GUI https://github.com/xgqfrms/gui/ https://www.npmjs.com/package/@xgqf ...
- cursor CSS属性定义鼠标指针悬浮在元素上时的外观。
1 1 cursor CSS属性定义鼠标指针悬浮在元素上时的外观. https://developer.mozilla.org/zh-CN/docs/Web/CSS/cursor 概述 cursor ...
- js & array & shuffle
js & array & shuffle const list = [1, 2, 3, 4, 5, 6, 7, 8, 9]; list.sort(() => Math.rando ...
- taro & Block
taro & Block https://nervjs.github.io/taro/docs/children.html#注意事项-1 import Taro, { Component, E ...
- Spring系列.Resource接口
接口简介 JDK中提供了java.net.URL这个类来用于获取不同种类的资源(根据不同前缀的url可以获取不同种类的资源).但是URL这个类没有获取classpath和ServletContext下 ...

