zoukankan      html  css  js  c++  java
  • BZOJ 1030 【JSOI2007】 文本生成器

    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;
    }
  • 相关阅读:
    冒泡/快速排序
    Windows RT和WinRT
    UAC(User Access Control)操作小结(C++)
    将CHM文件转换为HTML文件
    WPF实现窗口比例恒定不变小结(2)
    用WinForm的ShowHelp()函数处理CHM文件
    Windows 8下对Microsoft Surface SDK 2.0的调查小结
    WPF实现窗口比例恒定不变小结(1)
    资源释放
    axis 1.4 使用及介绍
  • 原文地址:https://www.cnblogs.com/lcf-2000/p/6009551.html
Copyright © 2011-2022 走看看