洛谷 P4112 [HEOI2015]最短不公共子串 解题报告
P4112 [HEOI2015]最短不公共子串
题目描述
在虐各种最长公共子串、子序列的题虐的不耐烦了之后,你决定反其道而行之。
一个串的“子串”指的是它的连续的一段,例如bcd是abcdef的子串,但bde不是。
一个串的“子序列”指的是它的可以不连续的一段,例如bde是abcdef的子串,但bdd不是。
下面,给两个小写字母串A,B,请你计算:
(1) A的一个最短的子串,它不是B的子串
(2) A的一个最短的子串,它不是B的子序列
(3) A的一个最短的子序列,它不是B的子串
(4) A的一个最短的子序列,它不是B的子序列
输入输出格式
输入格式:
有两行,每行一个小写字母组成的字符串,分别代表A和B。
输出格式:
输出4行,每行一个整数,表示以上4个问题的答案的长度。如果没有符合要求的答案,输出-1
说明
对于20%的数据,A和B的长度都不超过20
对于50%的数据,A和B的长度都不超过500
对于100%的数据,A和B的长度都不超过2000
四合一还行。
问题1
建立B的后缀自动机
枚举A的开头,然后在B上从S向后跑,失配了更新答案
问题2
建立B的序列自动机
然后一样的
问题3
在B的后缀自动机上跑DP(这样才能处理A的子序列)
具体的\(dp_i\)表示这个点可以匹配上的最短长度
从头开始枚举每个点更新。
问题4
在B的序列自动机上跑,其他和三一样
数据太水,3,4的dp01背包写成完全背包也可以过...
我3的dp写的是假的,懒得改了..
Code:
#include <cstdio>
#include <cstring>
const int N=4010;
const int inf=0x3f3f3f3f;
char s1[N],s2[N];
int n,dp[N];
int min(int x,int y){return x<y?x:y;}
struct SuffixAutomata
{
int tot,las,len[N],ch[N][26],par[N];
SuffixAutomata()
{
tot=las=1;
memset(len,0,sizeof(len));
memset(ch,0,sizeof(ch));
memset(par,0,sizeof(par));
}
void extend(int c)
{
int now=++tot,p=las;
len[now]=len[p]+1;
while(p&&!ch[p][c]) ch[p][c]=now,p=par[p];
if(!p) par[now]=1;
else
{
int x=ch[p][c];
if(len[x]==len[p]+1) par[now]=x;
else
{
int y=++tot;
len[y]=len[p]+1,par[y]=par[x];
memcpy(ch[y],ch[x],sizeof ch[y]);
while(p&&ch[p][c]==x) ch[p][c]=y,p=par[p];
par[now]=par[x]=y;
}
}
las=now;
}
void build(char *s)
{
int n=strlen(s+1);
for(int i=1;i<=n;i++) extend(s[i]-'a');
}
}T1;
struct SequentialAutomaton
{
int suc[N],ch[N][26];
SequentialAutomaton()
{
memset(suc,0,sizeof(suc));
memset(ch,0,sizeof(ch));
}
void build(char *s)
{
int n=strlen(s+1);s[0]='a';
for(int i=n;~i;i--)
{
for(int j=0;j<26;j++)
if(suc[j])
ch[i][j]=suc[j];
suc[s[i]-'a']=i;
}
}
}T2;
void work1()
{
int ans=inf;
for(int i=1;i<=n;i++)
{
int now=1;
for(int j=i;j<=n;j++)
{
int c=s1[j]-'a';
if(T1.ch[now][c]) now=T1.ch[now][c];
else
{
ans=min(ans,j+1-i);
break;
}
}
}
if(ans==inf) puts("-1");
else printf("%d\n",ans);
}
void work2()
{
int ans=inf;
for(int i=1;i<=n;i++)
{
int now=0;
for(int j=i;j<=n;j++)
{
int c=s1[j]-'a';
if(T2.ch[now][c]) now=T2.ch[now][c];
else
{
ans=min(ans,j+1-i);
break;
}
}
}
if(ans==inf) puts("-1");
else printf("%d\n",ans);
}
void work3()
{
memset(dp,0x3f,sizeof dp);
dp[1]=0;
int ans=inf;
for(int i=1;i<=n;i++)
{
int c=s1[i]-'a';
for(int j=1;j<=T1.tot;j++)
{
int v=T1.ch[j][c];
if(v) dp[v]=min(dp[v],dp[j]+1);
else ans=min(ans,dp[j]+1);
}
}
if(ans==inf) puts("-1");
else printf("%d\n",ans);
}
void work4()
{
memset(dp,0x3f,sizeof dp);
dp[0]=0;
int ans=inf;
for(int i=1;i<=n;i++)
{
int c=s1[i]-'a';
for(int j=n;~j;j--)
{
int v=T2.ch[j][c];
if(v) dp[v]=min(dp[v],dp[j]+1);
else ans=min(ans,dp[j]+1);
}
}
if(ans==inf) puts("-1");
else printf("%d\n",ans);
}
int main()
{
scanf("%s%s",s1+1,s2+1);
T1.build(s2),T2.build(s2),n=strlen(s1+1);
work1(),work2(),work3(),work4();
return 0;
}
2019.1.7
洛谷 P4112 [HEOI2015]最短不公共子串 解题报告的更多相关文章
- BZOJ 4032 Luogu P4112 [HEOI2015]最短不公共子串 (DP、后缀自动机)
这其实是道水题... 题目链接: (bzoj)https://www.lydsy.com/JudgeOnline/problem.php?id=4032 (luogu)https://www.luog ...
- 【BZOJ4032】[HEOI2015]最短不公共子串(后缀自动机,序列自动机)
[BZOJ4032][HEOI2015]最短不公共子串(后缀自动机,序列自动机) 题面 BZOJ 洛谷 题解 数据范围很小,直接暴力构建后缀自动机和序列自动机,然后直接在两个自动机上进行\(bfs\) ...
- BZOJ 4032: [HEOI2015]最短不公共子串
4032: [HEOI2015]最短不公共子串 Time Limit: 10 Sec Memory Limit: 256 MBSubmit: 446 Solved: 224[Submit][Sta ...
- BZOJ 4032: [HEOI2015]最短不公共子串 后缀自动机 暴力
4032: [HEOI2015]最短不公共子串 题目连接: http://www.lydsy.com/JudgeOnline/problem.php?id=4032 Description 在虐各种最 ...
- bzoj4032: [HEOI2015]最短不公共子串(SAM+DP)
4032: [HEOI2015]最短不公共子串 题目:传送门 题解: 陈年老题良心%你赛膜爆嘎爷 当初做题...一眼SAM...结果只会两种直接DP的情况... 情况1: 直接设f[i][j] 表示的 ...
- luoguP4112 [HEOI2015]最短不公共子串 SAM,序列自动机,广搜BFS
luoguP4112 [HEOI2015]最短不公共子串 链接 luogu loj 思路 子串可以用后缀自动机,子序列可以用序列自动机. 序列自动机是啥,就是能访问到所有子序列的自动机. 每个点记录下 ...
- bzoj4032/luoguP4112 [HEOI2015]最短不公共子串(后缀自动机+序列自动机上dp)
bzoj4032/luoguP4112 [HEOI2015]最短不公共子串(后缀自动机+序列自动机上dp) bzoj Luogu 题解时间 给两个小写字母串 $ A $ , $ B $ ,请你计算: ...
- BZOJ4032[HEOI2015]最短不公共子串——序列自动机+后缀自动机+DP+贪心
题目描述 在虐各种最长公共子串.子序列的题虐的不耐烦了之后,你决定反其道而行之. 一个串的“子串”指的是它的连续的一段,例如bcd是abcdef的子串,但bde不是. 一个串的“子序列”指的是它的可以 ...
- BZOJ4032:[HEOI2015]最短不公共子串(SAM)
Description 在虐各种最长公共子串.子序列的题虐的不耐烦了之后,你决定反其道而行之. 一个串的“子串”指的是它的连续的一段,例如bcd是abcdef的子串,但bde不是. 一个串的“子序列” ...
随机推荐
- # 2017-2018-2 20155319『网络对抗技术』Exp7:网络欺诈防范
2017-2018-2 20155319『网络对抗技术』Exp7:网络欺诈防范 一.原理与实践说明 1.实践目标 本实践的目标是:理解常用网络欺诈背后的原理,以提高防范意识,并提出具体防范方法. 2. ...
- MFC如何为程序添加图标
1.找几幅Ico格式的图片,可以在电脑中查找.ico一般都会找到.然后将ico文件放在工程目录下的res文件夹下. 2.点击菜单栏->编辑->添加资源->导入,选择res文件夹中将要 ...
- 变量内存空间的释放---c语言
堆栈内存释放: 栈的内存是由编译器自动分配.释放,出了作用域就释放. 堆的内存由程序员分配.释放,他的作用域是整个程序,如果程序没有释放,程序结束时会自动释放.
- [arc076F]Exhausted?[霍尔定理+线段树]
题意 地上 \(1\) 到 \(m\) 个位置摆上椅子,有 \(n\) 个人要就座,每个人都有座位癖好:选择 \(\le L\) 或者 \(\ge R\) 的位置.问至少需要在两边添加多少个椅子能让所 ...
- Asp.Net_from标签中的Enctype=multipart/form-data作用
ENCTYPE="multipart/form-data"用于表单里有图片上传. <form name="userInfo" method="p ...
- Centos 7 部署Kubernetes(K8S)集群
资源链接:https://pan.baidu.com/s/1-PT_QQAf7cTu_znX-S-r9Q 密码:33sr 转发:http://blog.51cto.com/lizhenliang/19 ...
- 这可能是最详细的Python文件操作
删除 # ==================删除==================# 只能删除文件,若为目录则报错# 若文件正在使用,Windows下会直接报错,Linux下会在目录表中删除记录, ...
- Github pages + Minimal-Mistakes + Disqus建立个人博客记录
本文详细记录了利用Github pages建立个人博客的步骤. github pages官方推荐使用Jekyll生成静态网页,jekyll支持各种不同的主题,Minimal-Mistakes是一个功能 ...
- Hyperledger Fabric网络节点架构
Fabric区块链网络的组成  区块链网络结构图 区块链网络组成 组成区块链网络相关的节点 节点是区块链的通信主体,和区块链网络相关的节点有多种类型:客户端(应用).Peer节点.排序服务(Orde ...
- 绕过用编码方式阻止XSS攻击的几个例子
阻止攻击的常用方法是:在将HTML返回给Web浏览器之前,对攻击者输入的HTML进行编码.HTML编码使用一些没有特定HTML意义的字符来代替那些标记字符(如尖括号).这些替代字符不会影响文本在web ...