您的位置:首页 > Web前端 > JavaScript

BZOJ4327 : JSOI2012 玄武密码

2015-11-21 01:35 537 查看
对所有询问串建立AC自动机。

然后将母串在AC自动机上跑,每走到一个点x,从x点出发沿着fail指针能到的所有前缀都是匹配成功的,暴力向上走,碰到走过的就break,这样每个点最多只会被标记一次。

时间复杂度$O(N+100M)$。

#include<cstdio>
#include<cstring>
const int N=10000003,M=100010;
int tot,son
[4],f
,fail
,q
;bool v
;
int n,m,i,fin[M],len[M];char a
,b[103];
inline int id(char x){
if(x=='S')return 0;
if(x=='E')return 1;
if(x=='W')return 2;
return 3;
}
inline void insert(int p){
int l=len[p]=strlen(b),x=0,i=0,w;
for(;i<l;x=son[x][w],i++)if(!son[x][w=id(b[i])])f[son[x][w]=++tot]=x;
fin[p]=x;
}
void make(){
int h=1,t=0,i,x;fail[0]=-1;
for(i=0;i<4;i++)if(son[0][i])q[++t]=son[0][i];
while(h<=t)for(x=q[h++],i=0;i<4;i++)if(son[x][i])fail[q[++t]=son[x][i]]=son[fail[x]][i];else son[x][i]=son[fail[x]][i];
}
void solve(){
for(int x=0,i=0,w;i<n;i++){
x=son[x][w=id(a[i])];
for(int y=x;~y;y=fail[y])if(v[y])break;else v[y]=1;
}
}
inline int ask(int p){
for(int x=fin[p],y=len[p];y;y--,x=f[x])if(v[x])return y;
return 0;
}
int main(){
scanf("%d%d%s",&n,&m,a);
for(i=0;i<m;i++)scanf("%s",b),insert(i);
make();
solve();
for(i=0;i<m;i++)printf("%d\n",ask(i));
return 0;
}


  
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: