给你若干给字符串,再给你一个m,问长度是m的字符串中包含给定字符串的数量mod 10007是多少

这个拿过来啥思路也没有,后来还是看了题解,才知道,原来,原来。。。。那个带fail的Trie还可以搞别的

网上的大多数都是26^m-补集做的,麻烦啊,网上有一个非洲猴的blog写的挺好的

设f[i][j][k]表示什么呢,表示i:0..1 j:0..m k:0...size

表示i状态,匹配到j,在Trie上是k的数量

f[i | val[next[c]]][j + 1][next[c]] += f[i][j][k];对就是这样

我已经把next[c] == NULL的都变成fail了,

提醒按照lrj的白书学的AC自动机的同学们,那个last数组在匹配的所有时候都要循环枚举

提醒我自己当k等于0是不是这个位置是无,而是一个未匹配的点,因为计算last和dp时用last耗时代码量费啊

#include <cstdio>
#include <cstdlib>
#include <cstring>
#include <queue>
#include <algorithm>
using namespace std;
struct AC{
    int ch[100010][30];
    int val[100010];
    char s[100010];
    int f[100010];
    int last[100010];
    int g[2][200][10010];
    int size;
    int len;
    inline int num(char x){
        return x - 'A';
    }
    inline void init(){
        memset(ch, 0, sizeof(ch));
        memset(val, 0, sizeof(val));
        size = 0;
        return;
    }
    inline void insert(){
        int u = 0;
        for(int i = 0; i < len; i ++){
            int c = num(s[i]);
            if(!ch[u][c]){
                ch[u][c] = ++ size;
            }
            u = ch[u][c];
        }
        val[u] = 1;
        return;
    }
    inline void getfail(){
        queue<int> Q;
        for(int i = 0; i < 26; i ++) {
            if(ch[0][i]) {
                Q.push(ch[0][i]);
            }
        }
        while(!Q.empty()){
            int r = Q.front(); Q.pop();
            for(int i = 0; i < 26; i ++){
                int u = ch[r][i];
                if(!u){
                    ch[r][i] = ch[f[r]][i];
                    continue;
                }
                Q.push(u);
                int v = f[r];
                while(v && !ch[v][i]) v = f[v];
                f[u] = ch[v][i];
            }
        }
        return;
    }
    inline int dp(int m){
        for(int i = 0; i <= size; i ++){
            if(!val[i]){
                int j = f[i];
                while(j){
                    if(val[j] == 1){
                        val[i] = 1;
                        break;
                    }
                    j = f[j];
                }
            }
        }
        g[0][0][0] = 1;
        for(int i = 0; i <= 1; i ++){
            for(int j = 0; j < m; j ++){
                for(int k = 0; k <= size; k ++){
                    for(int c = 0; c < 26; c ++){
                        g[i | val[ch[k][c]]][j + 1][ch[k][c]] += g[i][j][k];
                        g[i | val[ch[k][c]]][j + 1][ch[k][c]] %= 10007;
                    }
                }
            }
        }
        int ret = 0;
        for(int k = 0; k <= size; k ++) {
            ret += g[1][m][k];
            ret %= 10007;
        }
        return ret;
    }
} wt;
int main(){
    int N, M;
    scanf("%d%d", &N, &M);
    wt.init();
    for(int i = 1; i <= N; i ++){
        scanf("%s", wt.s);
        wt.len = strlen(wt.s);
        //if(wt.len > M) continue;
        wt.insert();
    }
    wt.getfail();
    printf("%d\n", wt.dp(M));
    return 0;
}

BZOJ1030——文本生成器的更多相关文章

  1. bzoj1030 文本生成器(AC自动机+dp)

    1030: [JSOI2007]文本生成器 Time Limit: 1 Sec  Memory Limit: 162 MBSubmit: 4777  Solved: 1986[Submit][Stat ...

  2. bzoj1030 文本生成器

    题目链接 题意 给出\(n\)个字符串,要构造一个长度为\(m\)的字符串\(S\),使得给出的\(n\)个字符串中至少有一个是\(S\)的子串.问方案数. 思路 \(AC\)自动机+\(DP\) 考 ...

  3. 【BZOJ1030】文本生成器(AC自动机,动态规划)

    [BZOJ1030]文本生成器(AC自动机,动态规划) 题面 BZOJ 题解 超级简单良心送分题 很明显是所有状态-不合法状态 合法状态就是\(26^m\) 不合法状态做一个\(dp\)就好 #inc ...

  4. 【BZOJ1030】[JSOI2007]文本生成器 AC自动机+动态规划

    [BZOJ1030][JSOI2007]文本生成器 Description JSOI交给队员ZYX一个任务,编制一个称之为“文本生成器”的电脑软件:该软件的使用者是一些低幼人群,他们现在使用的是GW文 ...

  5. BZOJ1030 JSOI2007 文本生成器 【AC自动机】【DP】*

    BZOJ1030 JSOI2007 文本生成器 Description JSOI交给队员ZYX一个任务,编制一个称之为"文本生成器"的电脑软件:该软件的使用者是一些低幼人群,他们现 ...

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

    [bzoj1030]: [JSOI2007]文本生成器 首先把匹配任意一个的个数的问题转化为总个数-没有一个匹配的个数 先构造AC自动机,然后枚举每一位的字母以及在自动机上的位置 f[i][j]为第i ...

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

    1030: [JSOI2007]文本生成器 Time Limit: 1 Sec  Memory Limit: 162 MBSubmit: 3253  Solved: 1330[Submit][Stat ...

  8. bzoj1030 [JSOI2007]文本生成器

    1030: [JSOI2007]文本生成器 Time Limit: 1 Sec  Memory Limit: 162 MBSubmit: 2654  Solved: 1100[Submit][Stat ...

  9. C++之路进阶——bzoj1030(文本生成器)

    F.A.Qs Home Discuss ProblemSet Status Ranklist Contest ModifyUser  hyxzc Logout 捐赠本站 Notice:由于本OJ建立在 ...

随机推荐

  1. 理解和使用 JavaScript 中的回调函数

    理解和使用 JavaScript 中的回调函数 标签: 回调函数指针js 2014-11-25 01:20 11506人阅读 评论(4) 收藏 举报  分类: JavaScript(4)    目录( ...

  2. 会话控制:session与cookie

    我们在浏览网站时,访问的每一个web页面都需要使用"http协议"实现.而HTTP协议是无状态协议,就是说HTTP协议没有一个内建机制来维护两个事务之间的状态.当一个用户请求一个页 ...

  3. 自然语言15.1_Part of Speech Tagging 词性标注

    QQ:231469242 欢迎喜欢nltk朋友交流 https://en.wikipedia.org/wiki/Part-of-speech_tagging In corpus linguistics ...

  4. 自然语言14.1_python实现PorterStemmer算法

    QQ:231469242 欢迎喜欢nltk朋友交流 #https://tartarus.org/martin/PorterStemmer/python.txt #!/usr/bin/env pytho ...

  5. 第1章 JavaScript简介

    概述:Javascript是一种依赖于网页浏览器的脚本语言.是一种脚本语言.由Netscape和Sun共同开发.与Java没有什么关系.作为一种语言标准,而被称为ECMACcript.一个JS的实现包 ...

  6. ecshop 订单-》设置默认收货地址,或者删除

    设置位置:ecs_users标的   country字段,默认是0,默认地址是 users_address 的address_id 设置默认收货地址 /** * 设置默认地址 * * @access ...

  7. Web启动服务器上的某一个服务

    情景是这样的.. 网页打开一个数据列表..数据要求实时从其他多个平台上获取.. 所以就有了一个Web页面..还有个WinService的服务来定时获取这些数据... 问题来了.. 发现Service有 ...

  8. 【struts2】Result和ResultType

    简单的说,Result是Action执行完后返回的一个字符串,它指示了Action执行完成后,下一个页面在哪里.Result仅仅是个字符串,仅仅是用来指示下一个页面的,那么如何才能够到达下一个页面呢? ...

  9. mysql SELECT FOR UPDATE语句使用示例

    以MySQL 的InnoDB 为例,预设的Tansaction isolation level 为REPEATABLE READ,在SELECT 的读取锁定主要分为两种方式:SELECT ... LO ...

  10. iOS- storyboard this class is not key value coding-compliant for the key xxx

    如图: 在使用storyboard的时候出现此问题,主要是因为给storybroad中的view拖线的时候,有时不小心线拖错了,或者再次拖线导致代码中控件的名字与之前拖线时定义的名字不同导致的. 解决 ...