【bzoj1030】: [JSOI2007]文本生成器 字符串-AC自动机-DP

【bzoj1030】: [JSOI2007]文本生成器

首先把匹配任意一个的个数的问题转化为总个数-没有一个匹配的个数

先构造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 }

时间: 2024-12-09 12:45:46

【bzoj1030】: [JSOI2007]文本生成器 字符串-AC自动机-DP的相关文章

bzoj1030: [JSOI2007]文本生成器(AC自动机+DP)

第一次写这类题...懵 直接计算答案不好计算,所以补集转化求不合法的方案. 首先考虑朴素的DP,设$f(i, s)$表示前$i$个字符,字符串为$s$的方案数,且任意一个给定串都不存在$s$中. 我们知道在一个字符串里找其他的字符串是AC自动机的强项,那么我们就可以考虑在AC自动机上跑DP,每次$+j$都在AC自动机上匹配,如果匹配到单词结尾的话就不能转移,否则就是可以转移的. 所以设$f(i, j)$为前$i$个字符,当前匹配到AC自动机上第$j$个节点的方案数,如果沿着fail一直往上的所有

bzoj 1030 [JSOI2007]文本生成器(AC自动机+DP)

1030: [JSOI2007]文本生成器 Time Limit: 1 Sec  Memory Limit: 162 MBSubmit: 3059  Solved: 1255[Submit][Status][Discuss] Description JSOI交给队员ZYX一个任务,编制一个称之为“文本生成器”的电脑软件:该软件的使用者是一些低幼人群,他们现在使用的是GW文本生成器v6版.该软件可以随机生成一些文章―――总是生成一篇长度固定且完全随机的文章—— 也就是说,生成的文章中每个字节都是完

[JSOI2007]文本生成器(AC自动机+DP)

题意 给你n个串.问有多少长度为m的串使得这n个串至少在其中出现过一次.输出答案膜10007意义下的结果. (n<=100,每个串的长度<=100) 题解 在AC自动机上跑DP. 用到一个容斥的思想,求至少出现过一次的次数就是,全部可能-一次都没出现的次数. 所以考虑dp,对于一个长度为i的串从i-1转移,i位置上的数必须不是n个串中某个串的结尾. 所以建立AC自动机,在上面跑.一个儿子从父亲转移时必须保证,儿子节点和儿子节点通过fail指针间接到达的点,都不是终止节点. 具体看代码. 1 #

BZOJ 1030 [JSOI2007]文本生成器(AC自动机)

[题目链接] http://www.lydsy.com/JudgeOnline/problem.php?id=1030 [题目大意] 求出包含任意一个给定串的串数量 [题解] 我们求出不包含任意一个给定串的数量,用全集去减即可, 对于给定串建立AC自动机,用1节点作为根,0节点向1连全字符集转移作为超级源, 那么0->match能匹配所有不包含给定串的串, 记dp[i][j]表示匹配了i长度,匹配到AC自动机j节点的串数量, 统计之后取补集即可. [代码] #include <cstdio&g

bzoj1030 [JSOI2007]文本生成器——AC自动机+DP

题目:https://www.lydsy.com/JudgeOnline/problem.php?id=1030 求至少有一个单词的文本串不太好求,所以转化成求所有情况减去没有一个单词的文本串: 没有一个单词的文本串可以用AC自动机+DP求,设 f[i][j] 表示文本串长度为 i ,当前 Trie 树上节点为 j 的方案数: 则 f[i][j] 可以转到仍然不包含单词的它的儿子的方案数中,同时文本串长度+1: 所以需要在 getfail 时把单词结尾的属性也转移一下,因为 fail 是单词结尾

BZOJ1030:[JSOI2007]文本生成器

1030: [JSOI2007]文本生成器 Time Limit: 1 Sec  Memory Limit: 162 MBSubmit: 4859  Solved: 2019[Submit][Status][Discuss] Description JSOI交给队员ZGX一个任务,编制一个称之为"文本生成器"的电脑软件:该软件的使用者是一些战舰狗,他们现在使用的是GW文本生成器v6版.该软件可以随机生成一些文章―――总是生成一篇长度固定且完全随机的文章-- 也就是说,生成的文章中每个字

[BZOJ1030] [JSOI2007] 文本生成器 (AC自动机 &amp; dp)

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

BZOJ1030 JSOI2007文本生成器

AC自动机. f[i][j]表示为文本串中长度为i匹配字典树中j号节点的方案数. 1 #include<bits/stdc++.h> 2 using namespace std; 3 typedef long long ll; 4 const int N=6005,mod=10007; 5 ll ans1,ans2,f[105][N]; 6 char s[105];int cnt,n,m; 7 queue<int>q; 8 struct node 9 { 10 int s,v[26

【BZOJ-1030】文本生成器 AC自动机 + DP

1030: [JSOI2007]文本生成器 Time Limit: 1 Sec  Memory Limit: 162 MBSubmit: 3253  Solved: 1330[Submit][Status][Discuss] Description JSOI交给队员ZYX一个任务,编制一个称之为“文本生成器”的电脑软件:该软件的使用者是一些低幼人群,他们现在使用的是GW文本生成器v6版.该软件可以随机生成一些文章―――总是生成一篇长度固定且完全随机的文章—— 也就是说,生成的文章中每个字节都是完