题目描述

在虐各种最长公共子串、子序列的题虐的不耐烦了之后,你决定反其道而行之。

一个串的“子串”指的是它的连续的一段,例如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.

输入输出样例

输入样例#1: 复制

aabbcc
abcabc
输出样例#1: 复制

2
4
2
4
输入样例#2: 复制

aabbcc
aabbcc
输出样例#2: 复制

-1
-1
2
-1

说明

对于20%的数据,A和B的长度都不超过20

对于50%的数据,A和B的长度都不超过500

对于100%的数据,A和B的长度都不超过2000

题解

  这题好烦……

  据说还有大佬是用dp的方法做的……相当于做了四道题……先膜为敬……

  然后看到有大佬是直接建上后缀自动机和回文自动机,然后在上面跑BFS,如果跑不动了就返回答案,可以保证最优

  然后我因为一堆sb错误调了一个多小时……

  为啥取队头元素不能直接整个结构体取啊!为啥我SAM都能打错啊!

 //minamoto
#include<cstring>
#include<algorithm>
#include<iostream>
#include<cstdio>
#include<queue>
using namespace std;
#define getc() (p1==p2&&(p2=(p1=buf)+fread(buf,1,1<<21,stdin),p1==p2)?EOF:*p1++)
char buf[<<],*p1=buf,*p2=buf;
char sr[<<],z[];int C=-,Z;
inline void Ot(){fwrite(sr,,C+,stdout),C=-;}
inline void print(int x){
if(C><<)Ot();if(x<)sr[++C]=,x=-x;
while(z[++Z]=x%+,x/=);
while(sr[++C]=z[Z],--Z);sr[++C]='\n';
}
const int N=;
struct SAM{
int fa[N],ch[N][],l[N];int last,cnt;
SAM(){last=cnt=;}
void ins(int c){
int p=last,np=++cnt;last=np,l[np]=l[p]+;
for(;p&&!ch[p][c];p=fa[p]) ch[p][c]=np;
if(!p) fa[np]=;
else{
int q=ch[p][c];
if(l[q]==l[p]+) fa[np]=q;
else{
int nq=++cnt;l[nq]=l[p]+;
memcpy(ch[nq],ch[q],sizeof(ch[q]));
fa[nq]=fa[q],fa[q]=fa[np]=nq;
for(;ch[p][c]==q;p=fa[p]) ch[p][c]=nq;
}
}
}
}SA,SB;
struct SqAM{
int ch[N][],lst[],fa[N];
int root,cnt;
SqAM(){root=cnt=;for(int i=;i<;++i) lst[i]=;}
void ins(int c){
int p=lst[c],np=++cnt;
fa[np]=p;
for(int i=;i<;++i)
for(int j=lst[i];j&&!ch[j][c];j=fa[j])
ch[j][c]=np;
lst[c]=np;
}
}SQA,SQB;
struct node{
int a,b,step;
node(int a=,int b=,int step=):a(a),b(b),step(step){}
};
int vis[N][N];
int bfs1(){
queue<node> q;
q.push(node(,,));
vis[][]=;
while(!q.empty()){
int ra=q.front().a,rb=q.front().b,st=q.front().step;
for(int i=;i<;i++){
int va=SA.ch[ra][i],vb=SB.ch[rb][i];
if(va&&vb){
if(vis[va][vb]==) continue;
q.push(node(va,vb,st+));
vis[va][vb]=;
}
if(va&&!vb) return st+;
}
q.pop();
}
return -;
}
int bfs2(){
queue<node> q;
q.push(node(,,));
vis[][]=;
while(!q.empty()){
int ra=q.front().a,rb=q.front().b,st=q.front().step;
for(int i=;i<;i++){
int va=SA.ch[ra][i],vb=SQB.ch[rb][i];
if(va&&vb){
if(vis[va][vb]==) continue;
q.push(node(va,vb,st+));
vis[va][vb]=;
}
if(va&&!vb) return st+;
}
q.pop();
}
return -;
}
int bfs3(){
queue<node> q;
q.push(node(,,));
vis[][]=;
while(!q.empty()){
int ra=q.front().a,rb=q.front().b,st=q.front().step;
for(int i=;i<;i++){
int va=SQA.ch[ra][i],vb=SB.ch[rb][i];
if(va&&vb){
if(vis[va][vb]==) continue;
q.push(node(va,vb,st+));
vis[va][vb]=;
}
if(va&&!vb) return st+;
}
q.pop();
}
return -;
}
int bfs4(){
queue<node> q;
q.push(node(,,));
vis[][]=;
while(!q.empty()){
int ra=q.front().a,rb=q.front().b,st=q.front().step;
for(int i=;i<;i++){
int va=SQA.ch[ra][i],vb=SQB.ch[rb][i];
if(va&&vb){
if(vis[va][vb]==) continue;
q.push(node(va,vb,st+));
vis[va][vb]=;
}
if(va&&!vb) return st+;
}
q.pop();
}
return -;
}
char a[N],b[N];int tot1,tot2;char ch;
int main(){
scanf("%s%s",a+,b+);
tot1=strlen(a+),tot2=strlen(b+);
for(int i=;i<=tot1;++i) SA.ins(a[i]-'a'),SQA.ins(a[i]-'a');
for(int i=;i<=tot2;++i) SB.ins(b[i]-'a'),SQB.ins(b[i]-'a');
printf("%d\n%d\n%d\n%d\n",bfs1(),bfs2(),bfs3(),bfs4());
return ;
}

BZOJ4032: [HEOI2015]最短不公共子串(后缀自动机+序列自动机)的更多相关文章

  1. bzoj4032/luoguP4112 [HEOI2015]最短不公共子串(后缀自动机+序列自动机上dp)

    bzoj4032/luoguP4112 [HEOI2015]最短不公共子串(后缀自动机+序列自动机上dp) bzoj Luogu 题解时间 给两个小写字母串 $ A $ , $ B $ ,请你计算: ...

  2. BZOJ 4032: [HEOI2015]最短不公共子串 后缀自动机 暴力

    4032: [HEOI2015]最短不公共子串 题目连接: http://www.lydsy.com/JudgeOnline/problem.php?id=4032 Description 在虐各种最 ...

  3. bzoj4032: [HEOI2015]最短不公共子串(SAM+DP)

    4032: [HEOI2015]最短不公共子串 题目:传送门 题解: 陈年老题良心%你赛膜爆嘎爷 当初做题...一眼SAM...结果只会两种直接DP的情况... 情况1: 直接设f[i][j] 表示的 ...

  4. BZOJ4032 [HEOI2015]最短不公共子串 【后缀自动机 + 序列自动机 + dp】

    题目链接 BZOJ4032 题解 首先膜\(hb\) 空手切神题 一问\(hash\),二问枚举 三问\(trie\)树,四问\(dp\) 南二巨佬神\(hb\) 空手吊打自动机 \(orz orz ...

  5. BZOJ4032[HEOI2015]最短不公共子串——序列自动机+后缀自动机+DP+贪心

    题目描述 在虐各种最长公共子串.子序列的题虐的不耐烦了之后,你决定反其道而行之. 一个串的“子串”指的是它的连续的一段,例如bcd是abcdef的子串,但bde不是. 一个串的“子序列”指的是它的可以 ...

  6. [BZOJ4032][HEOI2015]最短不公共子串(Trie+DP)

    在虐各种最长公共子串.子序列的题虐的不耐烦了之后,你决定反其道而行之——被它们虐. 操作一:对A,B分别建SAM,暴力BFS. 操作二:对B建序列自动机或SAM,A在上面暴力匹配. 操作三:对A,B建 ...

  7. BZOJ4032:[HEOI2015]最短不公共子串(SAM)

    Description 在虐各种最长公共子串.子序列的题虐的不耐烦了之后,你决定反其道而行之. 一个串的“子串”指的是它的连续的一段,例如bcd是abcdef的子串,但bde不是. 一个串的“子序列” ...

  8. bzoj 4032 [HEOI2015]最短不公共子串——后缀自动机

    题目:https://www.lydsy.com/JudgeOnline/problem.php?id=4032 不是 b 的子串的话就对 b 建后缀自动机,在 a 上枚举从每个位置开始的子串或者找子 ...

  9. BZOJ 4032: [HEOI2015]最短不公共子串(后缀自动机+记忆化搜索)

    传送门 解题思路 首先需要预处理两个串\(nxt(i)(j)\)表示i位置之后最近的\(j\). 第一问直接对\(b\)建后缀自动机,枚举\(a\)的起点暴力匹配. 第二问枚举\(a\)的起点,\(b ...

随机推荐

  1. C#的ComboBox学习使用2018.08.03

    ComboBox是一个有下拉列表的文本显示框,其text为当前的文本,item属性为项 comboBox1.Items.Add("); id = comboBox1.Text; 可以采用se ...

  2. MenuItem属性

    [MenuItem属性] The MenuItem attribute allows you to add menu items to the main menu. The MenuItem attr ...

  3. 【bzoj3667】Rabin-Miller算法

    3667: Rabin-Miller算法 Time Limit: 60 Sec  Memory Limit: 512 MBSubmit: 1200  Solved: 363[Submit][Statu ...

  4. OceanBase

    OceanBase 编辑 本词条缺少名片图,补充相关内容使词条更完整,还能快速升级,赶紧来编辑吧! OceanBase是一个支持海量数据的高性能分布式数据库系统,实现了 数千亿条记录.数百TB数据上的 ...

  5. 关于fastjson的一些知识

    今天被问到了一些有关fastjson的知识,问了fastjson内部的实现机制,笔者只是用过fastjson这个包,还真没了解过它的机制等. 下去后搜索了一些有关fastjson的知识,希望能对自己和 ...

  6. [C++] NEW Advanced Usage

    NEW Advanced Usage 将分配的内存限定在特定的一块区域 #include<iostream> #include<new> ); ); }; using name ...

  7. js失去焦点触发

    onblur="displayRest($(this))"

  8. Ubuntu 安装QT5 后编译程序报错: FindQt5Widgets.cmake

    安装QT5.4后,需要编译一个C++程序. Cmakelist 有find_package(Qt5Widgets REQUIRED),cmake 报错如下: CMake Error at CMakeL ...

  9. RTC实时时钟-备份区域BKP--原理讲解

    RTC(Real Time Clock):实时时钟 BCD码:用4位2进制来表示10以内的十进制的形式. RTC的时钟源:LSE(32.768KHZ).HSE_RTC.LSI.经过一个精密校准(RTC ...

  10. 反射机制:获取class的方法