【BZOJ4032】[HEOI2015]最短不公共子串(后缀自动机,序列自动机)
【BZOJ4032】[HEOI2015]最短不公共子串(后缀自动机,序列自动机)
题面
题解
数据范围很小,直接暴力构建后缀自动机和序列自动机,然后直接在两个自动机上进行\(bfs\),找到的第一个不同时存在的节点就直接输出就好了。
#include<iostream>
#include<cstdio>
#include<cstring>
#include<queue>
using namespace std;
#define MAX 4040
struct SAM
{
struct Node{int son[26],ff,len;}t[MAX];
int tot,last;void init(){tot=last=1;}
void extend(int c)
{
int p=last,np=++tot;last=tot;
t[np].len=t[p].len+1;
while(p&&!t[p].son[c])t[p].son[c]=np,p=t[p].ff;
if(!p)t[np].ff=1;
else
{
int q=t[p].son[c];
if(t[q].len==t[p].len+1)t[np].ff=q;
else
{
int nq=++tot;
t[nq]=t[q];t[nq].len=t[p].len+1;
t[q].ff=t[np].ff=nq;
while(p&&t[p].son[c]==q)t[p].son[c]=nq,p=t[p].ff;
}
}
}
}A,B;
struct SQAM
{
struct Node{int son[26];}t[MAX];
int lst[26],pre[MAX],tot,last;
void init(){tot=last=1;for(int i=0;i<26;++i)lst[i]=1;}
void extend(int c)
{
int p=lst[c],np=++tot;
pre[np]=p;
for(int i=0;i<26;++i)
for(int j=lst[i];j&&!t[j].son[c];j=pre[j])
t[j].son[c]=np;
lst[c]=np;
}
}AA,BB;
char SA[MAX],SB[MAX];
int na,nb;
struct Node{int len,a,b;};
bool vis[MAX][MAX];
int bfs1(SAM &A,SAM &B)
{
queue<Node> Q;Q.push((Node){0,1,1});
memset(vis,0,sizeof(vis));vis[1][1]=true;
while(!Q.empty())
{
Node p=Q.front();Q.pop();
for(int i=0;i<26;++i)
{
int x=A.t[p.a].son[i];
int y=B.t[p.b].son[i];
if(x&&y)
{
if(vis[x][y])continue;
vis[x][y]=true;Q.push((Node){p.len+1,x,y});
}
if(x&&!y)return p.len+1;
}
}
return -1;
}
int bfs2(SAM &A,SQAM &B)
{
queue<Node> Q;Q.push((Node){0,1,1});
memset(vis,0,sizeof(vis));vis[1][1]=true;
while(!Q.empty())
{
Node p=Q.front();Q.pop();
for(int i=0;i<26;++i)
{
int x=A.t[p.a].son[i];
int y=B.t[p.b].son[i];
if(x&&y)
{
if(vis[x][y])continue;
vis[x][y]=true;Q.push((Node){p.len+1,x,y});
}
if(x&&!y)return p.len+1;
}
}
return -1;
}
int bfs3(SQAM &A,SAM &B)
{
queue<Node> Q;Q.push((Node){0,1,1});
memset(vis,0,sizeof(vis));vis[1][1]=true;
while(!Q.empty())
{
Node p=Q.front();Q.pop();
for(int i=0;i<26;++i)
{
int x=A.t[p.a].son[i];
int y=B.t[p.b].son[i];
if(x&&y)
{
if(vis[x][y])continue;
vis[x][y]=true;Q.push((Node){p.len+1,x,y});
}
if(x&&!y)return p.len+1;
}
}
return -1;
}
int bfs4(SQAM &A,SQAM &B)
{
queue<Node> Q;Q.push((Node){0,1,1});
memset(vis,0,sizeof(vis));vis[1][1]=true;
while(!Q.empty())
{
Node p=Q.front();Q.pop();
for(int i=0;i<26;++i)
{
int x=A.t[p.a].son[i];
int y=B.t[p.b].son[i];
if(x&&y)
{
if(vis[x][y])continue;
vis[x][y]=true;Q.push((Node){p.len+1,x,y});
}
if(x&&!y)return p.len+1;
}
}
return -1;
}
int main()
{
scanf("%s%s",SA+1,SB+1);
na=strlen(SA+1);nb=strlen(SB+1);
A.init();B.init();AA.init();BB.init();
for(int i=1;i<=na;++i)A.extend(SA[i]-97);
for(int i=1;i<=nb;++i)B.extend(SB[i]-97);
for(int i=1;i<=na;++i)AA.extend(SA[i]-97);
for(int i=1;i<=nb;++i)BB.extend(SB[i]-97);
printf("%d\n%d\n%d\n%d\n",bfs1(A,B),bfs2(A,BB),bfs3(AA,B),bfs4(AA,BB));
return 0;
}
【BZOJ4032】[HEOI2015]最短不公共子串(后缀自动机,序列自动机)的更多相关文章
- bzoj4032/luoguP4112 [HEOI2015]最短不公共子串(后缀自动机+序列自动机上dp)
bzoj4032/luoguP4112 [HEOI2015]最短不公共子串(后缀自动机+序列自动机上dp) bzoj Luogu 题解时间 给两个小写字母串 $ A $ , $ B $ ,请你计算: ...
- 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] 表示的 ...
- BZOJ4032 [HEOI2015]最短不公共子串 【后缀自动机 + 序列自动机 + dp】
题目链接 BZOJ4032 题解 首先膜\(hb\) 空手切神题 一问\(hash\),二问枚举 三问\(trie\)树,四问\(dp\) 南二巨佬神\(hb\) 空手吊打自动机 \(orz orz ...
- BZOJ4032[HEOI2015]最短不公共子串——序列自动机+后缀自动机+DP+贪心
题目描述 在虐各种最长公共子串.子序列的题虐的不耐烦了之后,你决定反其道而行之. 一个串的“子串”指的是它的连续的一段,例如bcd是abcdef的子串,但bde不是. 一个串的“子序列”指的是它的可以 ...
- BZOJ4032: [HEOI2015]最短不公共子串(后缀自动机+序列自动机)
题目描述 在虐各种最长公共子串.子序列的题虐的不耐烦了之后,你决定反其道而行之. 一个串的“子串”指的是它的连续的一段,例如bcd是abcdef的子串,但bde不是. 一个串的“子序列”指的是它的可以 ...
- [BZOJ4032][HEOI2015]最短不公共子串(Trie+DP)
在虐各种最长公共子串.子序列的题虐的不耐烦了之后,你决定反其道而行之——被它们虐. 操作一:对A,B分别建SAM,暴力BFS. 操作二:对B建序列自动机或SAM,A在上面暴力匹配. 操作三:对A,B建 ...
- BZOJ4032:[HEOI2015]最短不公共子串(SAM)
Description 在虐各种最长公共子串.子序列的题虐的不耐烦了之后,你决定反其道而行之. 一个串的“子串”指的是它的连续的一段,例如bcd是abcdef的子串,但bde不是. 一个串的“子序列” ...
- bzoj 4032 [HEOI2015]最短不公共子串——后缀自动机
题目:https://www.lydsy.com/JudgeOnline/problem.php?id=4032 不是 b 的子串的话就对 b 建后缀自动机,在 a 上枚举从每个位置开始的子串或者找子 ...
- BZOJ 4032: [HEOI2015]最短不公共子串(后缀自动机+记忆化搜索)
传送门 解题思路 首先需要预处理两个串\(nxt(i)(j)\)表示i位置之后最近的\(j\). 第一问直接对\(b\)建后缀自动机,枚举\(a\)的起点暴力匹配. 第二问枚举\(a\)的起点,\(b ...
随机推荐
- 并发concurrent---3
背景:并发知识是一个程序员段位升级的体现,同样也是进入BAT的必经之路,有必要把并发知识重新梳理一遍. ConcurrentHashMap:在有了并发的基础知识以后,再来研究concurrent包.普 ...
- Django学习之七:Django 中间件
目录 Django 中间件 自定义中间件 - - - 大体两种方式 将中间件移除 实例 中间件加载源码阅读 总结 Django 中间件 Tips: 更新日志: 2019.01.31 更新django中 ...
- weblogic doc
BEA WebLogic Server 9.2 Documentation https://docs.oracle.com/cd/E13222_01/wls/docs92/index.html 8.1 ...
- Vue与React两个框架的区别对比
简单介绍 React--Facebook创建的JavaScript UI框架.它支撑着包括Instagram在内的大多数Facebook网站.React与当时流行的jQuery,Backbone.js ...
- (二)图数据neo4j基本认识
1.neo4j介绍 Neo4j是由Java和Scala实现的开源NoSQL图数据库.自2003年开始研发,直到2007年正式发布第一版.Neo4j的源代码托管在GitHub上,技术支持托管在Stack ...
- C# -- 使用 DriveInfo 获取磁盘驱动器信息
C# -- 使用 DriveInfo 获取磁盘驱动器信息 1. 代码实现 class Program { static void Main(string[] args) { GetComputerDi ...
- selenium之表格的定位
浏览器网页常常会包含各类表格,自动化测试工程师可能会经常操作表格中的行,列以及某些特定的单元格,因此熟练掌握表格的定位方法是自动化测试实施过程中必要的技能. 被测试网页的HTML代码 <!DOC ...
- vue源码分析—认识 Flow
认识 Flow Flow 是 facebook 出品的 JavaScript 静态类型检查⼯具.Vue.js 的源码利⽤了 Flow 做了静态类型检查, 所以了解 Flow 有助于我们阅读源码 Flo ...
- redis 安装时候遇到 jemalloc 问题记录
https://www.cnblogs.com/lovemdx/p/3199886.html https://blog.csdn.net/yfkiss/article/details/7035579 ...
- host头注入
看到有说这个题为出题而出题,其实我还是这么觉得, host出问题的话我觉得一般只有在审计代码,看到才知道有host注入 假设不提示host注入,就有难度了 常规的注入了