题目描述
题解
对于所有的模板串建立广义后缀自动机,对于每一个点统计一下right集合中有几个模板串
然后对于所有的查询串在后缀自动机上匹配,然后看一下最终匹配到的那个点有几个模板串就行了
具体的方法是记录一下每一个点最后匹配到的是哪个模板串,然后记录一个cnt,每一次暴力顺着pre指针往上跳,跳到已经更新过的就退出
时间复杂度不大好算,其实就是每一次加入一个串自动机上的点最多被遍历一次
感觉这题AC自动机也是很好做的,把所有的查询串扔到trie里建fail树,然后把模板串在自动机上匹配,每一次匹配到的所有点到根的路径上的所有查询串的end的答案都应该+1,这样区间修改单点查询;其实可以转化成单点修改然后转化为求子树权值和,并且因为每一次修改的点应该是一个并集,所以要按照dfs序排序然后容斥一下,不过是静态的最后只需要一次dfs就能出解了
本来还是想写一下的,但是这题先给的模板串再给的查询串,存起来比较麻烦,懒得写了
代码
#include<algorithm>
#include<iostream>
#include<cstring>
#include<cstdio>
#include<cmath>
using namespace std;
#define N 200005
char s[N];
int T,Q,n,root,sz,last,p,np,q,nq,ans;
int ch[N][130],pre[N],step[N],lc[N],cnt[N];
void extend(int id)
{
last=root;
for (int i=0;i<n;++i)
{
int x=s[i];
p=last;np=++sz;last=np;
step[np]=step[p]+1;
while (p&&!ch[p][x])
{
ch[p][x]=np;
p=pre[p];
}
if (!p) pre[np]=root;
else
{
q=ch[p][x];
if (step[q]==step[p]+1) pre[np]=q;
else
{
nq=++sz;
step[nq]=step[p]+1;
memcpy(ch[nq],ch[q],sizeof(ch[q]));
pre[nq]=pre[q];
lc[nq]=lc[q];cnt[nq]=cnt[q];
while (ch[p][x]==q)
{
ch[p][x]=nq;
p=pre[p];
}
pre[np]=pre[q]=nq;
}
}
p=np;
while (p&&lc[p]!=id)
{
lc[p]=id;
++cnt[p];
p=pre[p];
}
}
}
int sam()
{
p=root;
for (int i=0;i<n;++i)
{
int x=s[i];
if (!ch[p][x]) return 0;
else p=ch[p][x];
}
return cnt[p];
}
int main()
{
scanf("%d%d",&T,&Q);
root=++sz;
while (T)
{
scanf("%s",s);n=strlen(s);
extend(T);
--T;
}
while (Q--)
{
scanf("%s",s);n=strlen(s);
ans=sam();
printf("%d\n",ans);
}
}