首先把匹配任意一个的个数的问题转化为总个数-没有一个匹配的个数
先构造AC自动机,然后枚举每一位的字母以及在自动机上的位置
f[i][j]为第i位在j的位置且没有匹配过任何一个串的个数
然后26^m-sum(f[m][j])就是答案
还有就是当p->fail一直到root的路径上只要有一个点是一个串的终点那么点f[i][p]就要ban掉 因为这个WA了好多次
1 /* http://www.cnblogs.com/karl07/ */ 2 #include<cstdlib> 3 #include<cstdio> 4 #include<cstring> 5 #include<cmath> 6 #include<algorithm> 7 using namespace std; 8 9 #define P 10007 10 struct trie{ 11 trie *next[26],*fail; 12 int th,w; 13 }t[6005],*root=t,*NEW=t,*q[6005]; 14 15 int n,m,l,r,ans; 16 int f[105][6005]; 17 char s[105]; 18 19 trie *new1(){NEW++; NEW->th=NEW->w=0; return NEW;} 20 21 #define pnf p->next[i]->fail 22 #define pn p->next[i] 23 trie *insert(trie *p,int i,int w){ 24 if (!pn) pn=new1(); 25 pn->w|=w; 26 return pn; 27 } 28 29 void build_fail(){ 30 trie *p=q[0]=root; 31 for (int i=0;i<26;i++) if (pn) pnf=p,q[++r]=pn,pn->th=r; 32 while (l<r){ 33 p=q[++l]; 34 for (int i=0;i<26;i++){ 35 if (pn){ 36 q[++r]=pn,pn->th=r; 37 for (pnf=p->fail ; pnf!=root && !pnf->next[i] ; pnf=pnf->fail); 38 if (pnf->next[i]) pnf=pnf->next[i]; 39 pn->w|=pnf->w; 40 } 41 } 42 } 43 } 44 45 int Q_pow(int x,int y){ 46 int ans=1; 47 for (;y; x=x*x%P , y=y>>1 ) if (y&1) ans=ans*x%P; 48 return ans; 49 } 50 void dp(){ 51 f[0][0]=1; 52 for (int k=0;k<m;k++){ 53 for (int j=0;j<=r;j++) if (f[k][j] && !q[j]->w){ 54 for (int i=0;i<26;i++){ 55 trie *p=q[j]; 56 while (!pn && p!=root) p=p->fail; 57 if (pn) p=pn; 58 f[k+1][p->th]=(f[k+1][p->th]+f[k][j])%P; 59 } 60 } 61 } 62 for (int i=0;i<=r;i++) if (!q[i]->w) ans=(ans+f[m][i])%P; 63 printf("%d\n",(Q_pow(26,m)-ans+P)%P); 64 } 65 #undef pn 66 #undef pnf 67 68 int main(){ 69 scanf("%d%d",&n,&m); 70 for (int i=1;i<=n;i++){ 71 scanf("%s",s); 72 int j=0,l=strlen(s); 73 for (trie *p=root;j<l;j++) p=insert(p,s[j]-‘A‘,j==l-1); 74 } 75 build_fail(); 76 dp(); 77 return 0; 78 }
【bzoj1030】: [JSOI2007]文本生成器 字符串-AC自动机-DP
原文:http://www.cnblogs.com/karl07/p/6653397.html