BZOJ 4032: [HEOI2015]最短不公共子串(后缀自动机+记忆化搜索)
解题思路
首先需要预处理两个串$nxt(i)(j)$表示i位置之后最近的$j$。
第一问直接对$b$建后缀自动机,枚举$a$的起点暴力匹配。
第二问枚举$a$的起点,$b$用$nxt$跳。
第三问$a$与$b$一起跳,$b$用后缀自动机,$a$用$nxt$。
第四问$a$与$b$一起跳,都用$nxt$,要加记忆化。
代码
#include<iostream>
#include<cstdio>
#include<cstring>
#include<cmath>
#include<algorithm>
using namespace std;
const int N=2005;
inline int min(int x,int y) {return x<y?x:y;}
int lena,lenb,ans=1e9,nxta[N][27],prea[27],nxtb[N][27],preb[27];
int vis[N][N];
char A[N],B[N];
struct SAM{
int lst,cnt,fa[N<<1],ch[N<<1][27],l[N<<1];
inline void insert(int c){
int p=lst,np=++cnt; lst=cnt; l[np]=l[p]+1;
for(;p && !ch[p][c];p=fa[p]) ch[p][c]=np;
if(!p) fa[np]=1;
else {
int q=ch[p][c];
if(l[q]==l[p]+1) fa[np]=q;
else {
int nq=++cnt; l[nq]=l[p]+1;
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;
}
}
}
inline int query1(int x){
int p=1,ret=0;
for(int i=x;i<=lena;i++) {
if(ch[p][A[i]-'a']) p=ch[p][A[i]-'a'],ret++;
else return ret+1;
}
return 1e9;
}
inline void query3(int x,int y,int l){
for(int i=0;i<26;i++){
if(!ch[y][i] && nxta[x][i]) ans=min(ans,l);
else if(ch[y][i] && nxta[x][i]) query3(nxta[x][i],ch[y][i],l+1);
}
}
}sam;
inline int query2(int x){
int now=0,ret=0;
for(int i=x;i<=lena;i++){
if(!nxtb[now][A[i]-'a']) return ret+1;
ret++; now=nxtb[now][A[i]-'a'];
}
return 1e9;
}
int query4(int x,int y){
if(x && !y) return 1;
if(vis[x][y]) return vis[x][y];
int tmp=1e9;
for(int i=0;i<26;i++){
if(!nxta[x][i]) continue;
tmp=min(tmp,query4(nxta[x][i],nxtb[y][i]));
}
vis[x][y]=tmp+1;
return vis[x][y];
}
int main(){
scanf("%s%s",A+1,B+1); sam.cnt=sam.lst=1;
lena=strlen(A+1); lenb=strlen(B+1);
for(int i=lena;~i;i--) {
for(int j=0;j<26;j++) nxta[i][j]=prea[j];
prea[A[i]-'a']=i;
}
for(int i=lenb;~i;i--){
for(int j=0;j<26;j++) nxtb[i][j]=preb[j];
preb[B[i]-'a']=i;
}
for(int i=1;i<=lenb;i++) sam.insert(B[i]-'a');
for(int i=1;i<=lena;i++) ans=min(ans,sam.query1(i));
printf("%d\n",(ans==1e9)?-1:ans); ans=1e9;
for(int i=1;i<=lena;i++) ans=min(ans,query2(i));
printf("%d\n",(ans==1e9)?-1:ans); ans=1e9; sam.query3(0,1,0);
printf("%d\n",(ans==1e9)?-1:ans+1); ans=query4(0,0)-1;
printf("%d\n",(ans==1e9)?-1:ans);
return 0;
}
BZOJ 4032: [HEOI2015]最短不公共子串(后缀自动机+记忆化搜索)的更多相关文章
- BZOJ 4032: [HEOI2015]最短不公共子串 后缀自动机 暴力
4032: [HEOI2015]最短不公共子串 题目连接: http://www.lydsy.com/JudgeOnline/problem.php?id=4032 Description 在虐各种最 ...
- bzoj 4032 [HEOI2015]最短不公共子串——后缀自动机
题目:https://www.lydsy.com/JudgeOnline/problem.php?id=4032 不是 b 的子串的话就对 b 建后缀自动机,在 a 上枚举从每个位置开始的子串或者找子 ...
- bzoj4032/luoguP4112 [HEOI2015]最短不公共子串(后缀自动机+序列自动机上dp)
bzoj4032/luoguP4112 [HEOI2015]最短不公共子串(后缀自动机+序列自动机上dp) bzoj Luogu 题解时间 给两个小写字母串 $ A $ , $ B $ ,请你计算: ...
- BZOJ 4032: [HEOI2015]最短不公共子串
4032: [HEOI2015]最短不公共子串 Time Limit: 10 Sec Memory Limit: 256 MBSubmit: 446 Solved: 224[Submit][Sta ...
- BZOJ.4032.[HEOI2015]最短不公共子串(DP 后缀自动机)
题目链接 1.求A的最短子串,它不是B的子串. 子串是连续的,对B建SAM,枚举起点,在SAM上找到第一个无法匹配点即可.O(n)用SAM能做吗..开始想错了. 2.求A的最短子串,它不是B的子序列. ...
- bzoj 4032: [HEOI2015]最短不公共子串【dp+SAM】
第一.二问: 就是最小的最长公共长度+1,设f[i][j]为a匹配到i,b匹配到j,第一问的转移是f[i][j]=(a[i]==b[j]?f[i-1][j-1]+1:0),第二问的转移是f[i][j] ...
- BZOJ 4032: [HEOI2015]最短不公共子串 (dp*3 + SAM)
转博客大法好 第4个子任务中,为什么只转移最近的一个位置,自己YY吧(多YY有益身体健康). #include <bits/stdc++.h> using namespace std; t ...
- bzoj 4032 [ HEOI 2015 ] 最短不公共子串 —— 后缀自动机+序列自动机
题目:https://www.lydsy.com/JudgeOnline/problem.php?id=4032 序列自动机其实就是每个位置记录一下某字母后面第一个出现位置,为了子序列能尽量长. 对字 ...
- 【BZOJ】4032: [HEOI2015]最短不公共子串(LibreOJ #2123)
[题意]给两个小写字母串A,B,请你计算: (1) A的一个最短的子串,它不是B的子串 (2) A的一个最短的子串,它不是B的子序列 (3) A的一个最短的子序列,它不是B的子串 (4) A的一个最短 ...
随机推荐
- 【HDOJ6609】Find the answer(线段树)
题意:给定一个n个正整数的数列,第i项为w[i],对于每个i,你要从[1,i-1]中选择一些变成0,使得变化后[1,i]的总和小于m,每次询问最少要变几个 n<=2e5,m<=1e9,1& ...
- flutter Container组件和Text组件
在开始之前,我们先写一个最简单的入口文件: 后面,都是在这个结构的基础上面完成的. 由于Container组件和Text组件都是写在body里面的,所以下面,先将body抽离成一个组件的形式. ...
- CF 39E. What Has Dirichlet Got to Do with That?(记忆化搜索+博弈论)
传送门 解题思路 首先很好写出一个\(O(ab)\)的记搜,但发现这样无法处理\(a=1\)和\(b=1\)的情况,这两种情况需要特判.首先\(a=1\)的情况,就是如果当前选手让\(a+1\)必胜, ...
- AGC002 F Leftmost Ball——DP
题目:https://atcoder.jp/contests/agc002/tasks/agc002_f 充要条件是前缀0的个数 >= 颜色种数. 设计 DP ,放一个颜色的时候就把所有该颜色的 ...
- [CSP-S模拟测试]:God Knows(线段树维护单调栈)
题目描述 小$w$来到天堂的门口,对着天堂的大门发呆.大门上有一个二分图,左边第$i$个点连到右边第$p_i$个点.(保证$p_i$是一个排列).小$w$每次可以找左边某个对应连线尚未被移除的点$i$ ...
- [CSP-S模拟测试]:喝喝喝(模拟)
题目描述 奥利维尔和雪拉扎德在喝酒.两人连喝$18$瓶后,奥利维尔最终倒下了.奥利维尔服用了教会研究的醒酒药后,因为服用了太多产生了副作用,第二天睡不着了.他只好用数数的方式度过无聊的时光,不过他毕竟 ...
- 测开之路四十二:常用的jquery事件
$(‘selector’).click() 触发点击事件$(‘selector’).click(function) 添加点击事件$(‘selector’).dbclick() 触发双击事件$(‘sel ...
- 深入了解JAVA基础(面试)
I.常用类型与编码类问题: 1.Java中的基本类型有什么? byte.short.int.long.float.double.chart.boolean这八种,这 ...
- Python - zipfile 乱码问题解决
最近使用zipfile进行解包过程中遇到了很不舒服的问题,解包之后文件名是乱码的.下面进行简单总结: 首先,乱码肯定是因为解码方式不一样了,zipfile使用的是utf-8和cp437这两种编码方式, ...
- python 字典(dictionary)一些方法
1.python 字典(Dictionary) keys() 函数以列表返回一个字典所有的键. keys()语法: dict.keys() 2.setdefault()方法 python字典setde ...