BZOJ 1030 【JSOI2007】 文本生成器

时间:2023-12-19 19:57:26

Description

  JSOI交给队员ZYX一个任务,编制一个称之为“文本生成器”的电脑软件:该软件的使用者是一些低幼人群,
他们现在使用的是GW文本生成器v6版。该软件可以随机生成一些文章―――总是生成一篇长度固定且完全随机的文
章—— 也就是说,生成的文章中每个字节都是完全随机的。如果一篇文章中至少包含使用者们了解的一个单词,
那么我们说这篇文章是可读的(我们称文章a包含单词b,当且仅当单词b是文章a的子串)。但是,即使按照这样的
标准,使用者现在使用的GW文本生成器v6版所生成的文章也是几乎完全不可读的?。ZYX需要指出GW文本生成器 v6
生成的所有文本中可读文本的数量,以便能够成功获得v7更新版。你能帮助他吗?

Input

  输入文件的第一行包含两个正整数,分别是使用者了解的单词总数N (<= 60),GW文本生成器 v6生成的文本固
定长度M;以下N行,每一行包含一个使用者了解的单词。这里所有单词及文本的长度不会超过100,并且只可能包
含英文大写字母A..Z

Output

  一个整数,表示可能的文章总数。只需要知道结果模10007的值。

  像我这样成天刷水题吃枣药丸
  这道题就是问有多少个 包含至少一个给定串,并且长度为$m$的串。然后,显然这种多串处理的题是要AC自动机的。构出AC自动机后在上面dp就可以了。
  其实感觉直接dp也可以做,但其实把问题转化一下更好做。求出所有不合法的串然后用总数量减一下就可以了。于是$f_{i,j}$表示在AC自动机上走了$i$步后到达节点$j$的方案数,转移的时候不经过单词的结束节点即可。
  AC自动机都写错的我吃枣药丸
  下面贴代码:
#include<iostream>
#include<cstdio>
#include<cstring>
#include<algorithm>
#include<cmath>
#define File(s) freopen(s".in","r",stdin),freopen(s".out","w",stdout)
#define maxn 6010
#define mod 10007 using namespace std;
typedef long long llg; int n,m,fl[maxn],d[maxn],ans;
int ch[maxn][26],sz,f[101][maxn];
bool val[maxn],w[maxn]; int getint(){
int w=0;bool q=0;
char c=getchar();
while((c>'9'||c<'0')&&c!='-') c=getchar();
if(c=='-') c=getchar(),q=1;
while(c>='0'&&c<='9') w=w*10+c-'0',c=getchar();
return q?-w:w;
} void gi(int &x){if(x>=mod) x%=mod;}
void insert(){
char c=getchar();
while(c>'Z'||c<'A') c=getchar();
int u=0;
while(c>='A' && c<='Z'){
if(!ch[u][c-'A']) ch[u][c-'A']=++sz;
u=ch[u][c-'A']; c=getchar();
}
val[u]=1;
} void getf(){
int l=0,r=0,u;d[r++]=0;
while(l!=r){
u=d[l++];
for(int i=0,j;i<26;i++)
if(ch[u][i]){
j=fl[u];
while(!ch[j][i] && j) j=fl[j];
if(u!=j){
fl[ch[u][i]]=ch[j][i];
val[ch[u][i]]|=val[ch[j][i]];
}
d[r++]=ch[u][i];
}
else ch[u][i]=ch[fl[u]][i];
}
} int main(){
File("a");
n=getint(); m=getint();
while(n--) insert();
getf(); f[0][0]=ans=1;
for(int i=0;i<m;i++)
for(int u=0;u<=sz;u++)
if(!val[u])
for(int j=0;j<26;j++)
f[i+1][ch[u][j]]+=f[i][u],gi(f[i+1][ch[u][j]]);
for(int i=1;i<=m;i++) ans*=26,gi(ans);
for(int u=0;u<=sz;u++) if(!val[u]) ans-=f[m][u],ans+=mod,gi(ans);
printf("%d",ans);
return 0;
}