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

本文探讨了JSOI队员面临的任务,即计算GW文本生成器v6生成的可读文本数量,通过状态压缩和Trie树算法实现。文章介绍了算法的具体实现过程,包括构建Trie树、状态转移和记忆化搜索。

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的值。

Sample Input

2 2
A
B

Sample Output

100
 
 

这题直接推可读文本似乎状压可做

然后就被我码废了

然后我就不是人辽

得知ans=26m-完全不可读文本数量后这题就灰常套路了

跑建图 转移一下end标记 之后直接记忆化搜索实现dp

dp[i][j]表示还剩i位 转移到了Trie图的j节点

(我不会说自己把记忆化搜索传参写反 调了2h的QAQ)

#include<cstdio>
#include<iostream>
#include<cstring>
#include<queue>
using namespace std;
const int mod=10007;
char s[105];
int n,tot=1,m;
bool tag[5805];
struct trie
{
    int son[28],fail;
}t[5805];
void ins(char *str)
{
    int l=strlen(str+1),root=1;
    for(int i=1;i<=l;i++)
    {
        int x=str[i]-'A';
        if(!t[root].son[x])t[root].son[x]=++tot;
        root=t[root].son[x];
    }
    tag[root]=1;
}
void build()
{
    queue<int> q;
    for(int i=0;i<26;i++)t[0].son[i]=1;
    q.push(1);
    while(!q.empty())
    {
        int x=q.front();
        q.pop();
        for(int i=0;i<26;i++)
        {
            int &y=t[x].son[i];
            if(!y)
            {
                y=t[t[x].fail].son[i];
                continue;
            }
            t[y].fail=t[t[x].fail].son[i];
            q.push(y);
            if(tag[t[t[x].fail].son[i]])tag[t[x].son[i]]=1;
        }
    }
}
int dp[105][5805],v[105][5805];
int dfs(int x,int y)
{
    if(!x)return 1;
    if(v[x][y])return dp[x][y];
    v[x][y]=1;
    for(int i=0;i<26;i++)
        if(!tag[t[y].son[i]])(dp[x][y]+=dfs(x-1,t[y].son[i]))%=mod;
    return dp[x][y];
}
int main()
{
    scanf("%d%d",&n,&m);
    for(int i=1;i<=n;i++)
    {
        scanf("%s",s+1);
        ins(s);
    }
    build();
    int res=dfs(m,1),po=1;
    for(int i=1;i<=m;i++)po*=26,po%=mod;
    cout<<(po-res+mod)%mod<<endl;
    return 0;
}

 

转载于:https://www.cnblogs.com/Rorschach-XR/p/11081805.html

# P4052 [JSOI2007] 文本生成器 ## 题目描述 JSOI 交给队员 ZYX 一个任务,编制一个称之为“文本生成器”的电脑软件:该软件的使用者是一些低幼人群,他们现在使用的是 GW 文本生成器 v6 版。 该软件可以随机生成一些文章——总是生成一篇长度固定且完全随机的文章。 也就是说,生成的文章中每个字符都是完全随机的。如果一篇文章中至少包含使用者们了解的一个单词,那么我们说这篇文章是可读的(我们称文章 $s$ 包含单词 $t$,当且仅当单词 $t$ 是文章 $s$ 的子串)。但是,即使按照这样的标准,使用者现在使用的 GW 文本生成器 v6 版所生成的文章也是几乎完全不可读的。ZYX 需要指出 GW 文本生成器 v6 生成的所有文本中,可读文本的数量,以便能够成功获得 v7 更新版。你能帮助他吗? 答案对 $10^4 + 7$ 取模。 ## 输入格式 第一行有两个整数,分别表示使用者了解的单词总数 $n$ 和生成的文章长度 $m$。 接下来 $n$ 行,每行一个字符串 $s_i$,表示一个使用者了解的单词。 ## 输出格式 输出一行一个整数表示答案对 $10^4 + 7$ 取模的结果。 ## 输入输出样例 #1 ### 输入 #1 ``` 2 2 A B ``` ### 输出 #1 ``` 100 ``` ## 说明/提示 #### 数据规模与约定 对于全部的测试点,保证: - $1 \leq n \leq 60$,$1 \leq m \leq 100$。 - $1 \leq |s_i| \leq 100$,其中 $|s_i|$ 表示字符串 $s_i$ 的长度。 - $s_i$ 中只含大写英文字母。 为什么WA了(过了样例): ```cpp #include <bits/stdc++.h> using namespace std; const int MOD = 1e4 + 7; const int MAX_N = 100; const int LENGTH = 10050; int n, m, trie[LENGTH][26], tot; int fail[LENGTH], dp[MAX_N][LENGTH]; bool vis[LENGTH]; void insert(string s) { int cur = 0; for (char ch : s) { int i = ch - 'A'; if (!trie[cur][i]) trie[cur][i] = ++tot; cur = trie[cur][i]; } vis[cur] = true; } void build_fail() { queue<int> q; for (int i = 0; i < 26; i++) if (trie[0][i]) q.push(trie[0][i]); while (!q.empty()) { int cur = q.front(); q.pop(); vis[cur] |= vis[fail[cur]]; for (int i = 0; i < 26; i++) if (trie[cur][i]) { fail[trie[cur][i]] = trie[fail[cur]][i]; q.push(trie[cur][i]); } else trie[cur][i] = trie[fail[cur]][i]; } } int main() { cin >> n >> m; for (int i = 1; i <= m; i++) { string s; cin >> s; insert(s); } build_fail(); dp[0][0] = 1; for (int i = 0; i < n; i++) for (int j = 0; j <= tot; j++) for (int k = 0; k < 26; k++) if (!vis[trie[j][k]]) dp[i + 1][trie[j][k]] = (dp[i + 1][trie[j][k]] + dp[i][j]) % MOD; int ans = 1; for (int i = 1; i <= n; i++) ans = ans * 26 % MOD; for (int i = 0; i <= tot; i++) ans = (ans - dp[n][i]) % MOD; cout << (ans + MOD) % MOD << '\n'; return 0; } ```
最新发布
08-02
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值